1 /*
2 * Hewlett-Packard Page Control Language filter for CUPS.
3 *
4 * Copyright 2007-2015 by Apple Inc.
5 * Copyright 1993-2007 by Easy Software Products.
6 *
7 * Licensed under Apache License v2.0. See the file "LICENSE" for more information.
8 */
9
10 /*
11 * Include necessary headers...
12 */
13
14 #include <cups/cups.h>
15 #include <cups/ppd.h>
16 #include <cups/string-private.h>
17 #include <cups/language-private.h>
18 #include <cups/raster.h>
19 #include <unistd.h>
20 #include <fcntl.h>
21 #include <signal.h>
22
23
24 /*
25 * Globals...
26 */
27
28 unsigned char *Planes[4], /* Output buffers */
29 *CompBuffer, /* Compression buffer */
30 *BitBuffer; /* Buffer for output bits */
31 unsigned NumPlanes, /* Number of color planes */
32 ColorBits, /* Number of bits per color */
33 Feed; /* Number of lines to skip */
34 cups_bool_t Duplex; /* Current duplex mode */
35 int Page, /* Current page number */
36 Canceled; /* Has the current job been canceled? */
37
38
39 /*
40 * Prototypes...
41 */
42
43 void Setup(void);
44 void StartPage(ppd_file_t *ppd, cups_page_header2_t *header);
45 void EndPage(void);
46 void Shutdown(void);
47
48 void CancelJob(int sig);
49 void CompressData(unsigned char *line, unsigned length, unsigned plane, unsigned type);
50 void OutputLine(cups_page_header2_t *header);
51
52
53 /*
54 * 'Setup()' - Prepare the printer for printing.
55 */
56
57 void
Setup(void)58 Setup(void)
59 {
60 /*
61 * Send a PCL reset sequence.
62 */
63
64 putchar(0x1b);
65 putchar('E');
66 }
67
68
69 /*
70 * 'StartPage()' - Start a page of graphics.
71 */
72
73 void
StartPage(ppd_file_t * ppd,cups_page_header2_t * header)74 StartPage(ppd_file_t *ppd, /* I - PPD file */
75 cups_page_header2_t *header) /* I - Page header */
76 {
77 unsigned plane; /* Looping var */
78
79
80 /*
81 * Show page device dictionary...
82 */
83
84 fprintf(stderr, "DEBUG: StartPage...\n");
85 fprintf(stderr, "DEBUG: Duplex = %d\n", header->Duplex);
86 fprintf(stderr, "DEBUG: HWResolution = [ %d %d ]\n", header->HWResolution[0], header->HWResolution[1]);
87 fprintf(stderr, "DEBUG: ImagingBoundingBox = [ %d %d %d %d ]\n", header->ImagingBoundingBox[0], header->ImagingBoundingBox[1], header->ImagingBoundingBox[2], header->ImagingBoundingBox[3]);
88 fprintf(stderr, "DEBUG: Margins = [ %d %d ]\n", header->Margins[0], header->Margins[1]);
89 fprintf(stderr, "DEBUG: ManualFeed = %d\n", header->ManualFeed);
90 fprintf(stderr, "DEBUG: MediaPosition = %d\n", header->MediaPosition);
91 fprintf(stderr, "DEBUG: NumCopies = %d\n", header->NumCopies);
92 fprintf(stderr, "DEBUG: Orientation = %d\n", header->Orientation);
93 fprintf(stderr, "DEBUG: PageSize = [ %d %d ]\n", header->PageSize[0], header->PageSize[1]);
94 fprintf(stderr, "DEBUG: cupsWidth = %d\n", header->cupsWidth);
95 fprintf(stderr, "DEBUG: cupsHeight = %d\n", header->cupsHeight);
96 fprintf(stderr, "DEBUG: cupsMediaType = %d\n", header->cupsMediaType);
97 fprintf(stderr, "DEBUG: cupsBitsPerColor = %d\n", header->cupsBitsPerColor);
98 fprintf(stderr, "DEBUG: cupsBitsPerPixel = %d\n", header->cupsBitsPerPixel);
99 fprintf(stderr, "DEBUG: cupsBytesPerLine = %d\n", header->cupsBytesPerLine);
100 fprintf(stderr, "DEBUG: cupsColorOrder = %d\n", header->cupsColorOrder);
101 fprintf(stderr, "DEBUG: cupsColorSpace = %d\n", header->cupsColorSpace);
102 fprintf(stderr, "DEBUG: cupsCompression = %d\n", header->cupsCompression);
103
104 /*
105 * Setup printer/job attributes...
106 */
107
108 Duplex = header->Duplex;
109 ColorBits = header->cupsBitsPerColor;
110
111 if ((!Duplex || (Page & 1)) && header->MediaPosition)
112 printf("\033&l%dH", /* Set media position */
113 header->MediaPosition);
114
115 if (Duplex && ppd && ppd->model_number == 2)
116 {
117 /*
118 * Handle duplexing on new DeskJet printers...
119 */
120
121 printf("\033&l-2H"); /* Load media */
122
123 if (Page & 1)
124 printf("\033&l2S"); /* Set duplex mode */
125 }
126
127 if (!Duplex || (Page & 1) || (ppd && ppd->model_number == 2))
128 {
129 /*
130 * Set the media size...
131 */
132
133 printf("\033&l6D\033&k12H"); /* Set 6 LPI, 10 CPI */
134 printf("\033&l0O"); /* Set portrait orientation */
135
136 switch (header->PageSize[1])
137 {
138 case 540 : /* Monarch Envelope */
139 printf("\033&l80A"); /* Set page size */
140 break;
141
142 case 595 : /* A5 */
143 printf("\033&l25A"); /* Set page size */
144 break;
145
146 case 624 : /* DL Envelope */
147 printf("\033&l90A"); /* Set page size */
148 break;
149
150 case 649 : /* C5 Envelope */
151 printf("\033&l91A"); /* Set page size */
152 break;
153
154 case 684 : /* COM-10 Envelope */
155 printf("\033&l81A"); /* Set page size */
156 break;
157
158 case 709 : /* B5 Envelope */
159 printf("\033&l100A"); /* Set page size */
160 break;
161
162 case 756 : /* Executive */
163 printf("\033&l1A"); /* Set page size */
164 break;
165
166 case 792 : /* Letter */
167 printf("\033&l2A"); /* Set page size */
168 break;
169
170 case 842 : /* A4 */
171 printf("\033&l26A"); /* Set page size */
172 break;
173
174 case 1008 : /* Legal */
175 printf("\033&l3A"); /* Set page size */
176 break;
177
178 case 1191 : /* A3 */
179 printf("\033&l27A"); /* Set page size */
180 break;
181
182 case 1224 : /* Tabloid */
183 printf("\033&l6A"); /* Set page size */
184 break;
185 }
186
187 printf("\033&l%dP", /* Set page length */
188 header->PageSize[1] / 12);
189 printf("\033&l0E"); /* Set top margin to 0 */
190 }
191
192 if (!Duplex || (Page & 1))
193 {
194 /*
195 * Set other job options...
196 */
197
198 printf("\033&l%dX", header->NumCopies); /* Set number copies */
199
200 if (header->cupsMediaType &&
201 (!ppd || ppd->model_number != 2 || header->HWResolution[0] == 600))
202 printf("\033&l%dM", /* Set media type */
203 header->cupsMediaType);
204
205 if (!ppd || ppd->model_number != 2)
206 {
207 int mode = Duplex ? 1 + header->Tumble != 0 : 0;
208
209 printf("\033&l%dS", mode); /* Set duplex mode */
210 printf("\033&l0L"); /* Turn off perforation skip */
211 }
212 }
213 else if (!ppd || ppd->model_number != 2)
214 printf("\033&a2G"); /* Set back side */
215
216 /*
217 * Set graphics mode...
218 */
219
220 if (ppd && ppd->model_number == 2)
221 {
222 /*
223 * Figure out the number of color planes...
224 */
225
226 if (header->cupsColorSpace == CUPS_CSPACE_KCMY)
227 NumPlanes = 4;
228 else
229 NumPlanes = 1;
230
231 /*
232 * Set the resolution and top-of-form...
233 */
234
235 printf("\033&u%dD", header->HWResolution[0]);
236 /* Resolution */
237 printf("\033&l0e0L"); /* Reset top and don't skip */
238 printf("\033*p0Y\033*p0X"); /* Set top of form */
239
240 /*
241 * Send 26-byte configure image data command with horizontal and
242 * vertical resolutions as well as a color count...
243 */
244
245 printf("\033*g26W");
246 putchar(2); /* Format 2 */
247 putchar((int)NumPlanes); /* Output planes */
248
249 putchar((int)(header->HWResolution[0] >> 8));/* Black resolution */
250 putchar((int)header->HWResolution[0]);
251 putchar((int)(header->HWResolution[1] >> 8));
252 putchar((int)header->HWResolution[1]);
253 putchar(0);
254 putchar(1 << ColorBits); /* # of black levels */
255
256 putchar((int)(header->HWResolution[0] >> 8));/* Cyan resolution */
257 putchar((int)header->HWResolution[0]);
258 putchar((int)(header->HWResolution[1] >> 8));
259 putchar((int)header->HWResolution[1]);
260 putchar(0);
261 putchar(1 << ColorBits); /* # of cyan levels */
262
263 putchar((int)(header->HWResolution[0] >> 8));/* Magenta resolution */
264 putchar((int)header->HWResolution[0]);
265 putchar((int)(header->HWResolution[1] >> 8));
266 putchar((int)header->HWResolution[1]);
267 putchar(0);
268 putchar(1 << ColorBits); /* # of magenta levels */
269
270 putchar((int)(header->HWResolution[0] >> 8));/* Yellow resolution */
271 putchar((int)header->HWResolution[0]);
272 putchar((int)(header->HWResolution[1] >> 8));
273 putchar((int)header->HWResolution[1]);
274 putchar(0);
275 putchar(1 << ColorBits); /* # of yellow levels */
276
277 printf("\033&l0H"); /* Set media position */
278 }
279 else
280 {
281 printf("\033*t%uR", header->HWResolution[0]);
282 /* Set resolution */
283
284 if (header->cupsColorSpace == CUPS_CSPACE_KCMY)
285 {
286 NumPlanes = 4;
287 printf("\033*r-4U"); /* Set KCMY graphics */
288 }
289 else if (header->cupsColorSpace == CUPS_CSPACE_CMY)
290 {
291 NumPlanes = 3;
292 printf("\033*r-3U"); /* Set CMY graphics */
293 }
294 else
295 NumPlanes = 1; /* Black&white graphics */
296
297 /*
298 * Set size and position of graphics...
299 */
300
301 printf("\033*r%uS", header->cupsWidth); /* Set width */
302 printf("\033*r%uT", header->cupsHeight); /* Set height */
303
304 printf("\033&a0H"); /* Set horizontal position */
305
306 if (ppd)
307 printf("\033&a%.0fV", /* Set vertical position */
308 10.0 * (ppd->sizes[0].length - ppd->sizes[0].top));
309 else
310 printf("\033&a0V"); /* Set top-of-page */
311 }
312
313 printf("\033*r1A"); /* Start graphics */
314
315 if (header->cupsCompression)
316 printf("\033*b%uM", /* Set compression */
317 header->cupsCompression);
318
319 Feed = 0; /* No blank lines yet */
320
321 /*
322 * Allocate memory for a line of graphics...
323 */
324
325 if ((Planes[0] = malloc(header->cupsBytesPerLine + NumPlanes)) == NULL)
326 {
327 fputs("ERROR: Unable to allocate memory\n", stderr);
328 exit(1);
329 }
330
331 for (plane = 1; plane < NumPlanes; plane ++)
332 Planes[plane] = Planes[0] + plane * header->cupsBytesPerLine / NumPlanes;
333
334 if (ColorBits > 1)
335 BitBuffer = malloc(ColorBits * ((header->cupsWidth + 7) / 8));
336 else
337 BitBuffer = NULL;
338
339 if (header->cupsCompression)
340 CompBuffer = malloc(header->cupsBytesPerLine * 2 + 2);
341 else
342 CompBuffer = NULL;
343 }
344
345
346 /*
347 * 'EndPage()' - Finish a page of graphics.
348 */
349
350 void
EndPage(void)351 EndPage(void)
352 {
353 /*
354 * Eject the current page...
355 */
356
357 if (NumPlanes > 1)
358 {
359 printf("\033*rC"); /* End color GFX */
360
361 if (!(Duplex && (Page & 1)))
362 printf("\033&l0H"); /* Eject current page */
363 }
364 else
365 {
366 printf("\033*r0B"); /* End GFX */
367
368 if (!(Duplex && (Page & 1)))
369 printf("\014"); /* Eject current page */
370 }
371
372 fflush(stdout);
373
374 /*
375 * Free memory...
376 */
377
378 free(Planes[0]);
379
380 if (BitBuffer)
381 free(BitBuffer);
382
383 if (CompBuffer)
384 free(CompBuffer);
385 }
386
387
388 /*
389 * 'Shutdown()' - Shutdown the printer.
390 */
391
392 void
Shutdown(void)393 Shutdown(void)
394 {
395 /*
396 * Send a PCL reset sequence.
397 */
398
399 putchar(0x1b);
400 putchar('E');
401 }
402
403
404 /*
405 * 'CancelJob()' - Cancel the current job...
406 */
407
408 void
CancelJob(int sig)409 CancelJob(int sig) /* I - Signal */
410 {
411 (void)sig;
412
413 Canceled = 1;
414 }
415
416
417 /*
418 * 'CompressData()' - Compress a line of graphics.
419 */
420
421 void
CompressData(unsigned char * line,unsigned length,unsigned plane,unsigned type)422 CompressData(unsigned char *line, /* I - Data to compress */
423 unsigned length, /* I - Number of bytes */
424 unsigned plane, /* I - Color plane */
425 unsigned type) /* I - Type of compression */
426 {
427 unsigned char *line_ptr, /* Current byte pointer */
428 *line_end, /* End-of-line byte pointer */
429 *comp_ptr, /* Pointer into compression buffer */
430 *start; /* Start of compression sequence */
431 unsigned count; /* Count of bytes for output */
432
433
434 switch (type)
435 {
436 default :
437 /*
438 * Do no compression...
439 */
440
441 line_ptr = line;
442 line_end = line + length;
443 break;
444
445 case 1 :
446 /*
447 * Do run-length encoding...
448 */
449
450 line_end = line + length;
451 for (line_ptr = line, comp_ptr = CompBuffer;
452 line_ptr < line_end;
453 comp_ptr += 2, line_ptr += count)
454 {
455 for (count = 1;
456 (line_ptr + count) < line_end &&
457 line_ptr[0] == line_ptr[count] &&
458 count < 256;
459 count ++);
460
461 comp_ptr[0] = (unsigned char)(count - 1);
462 comp_ptr[1] = line_ptr[0];
463 }
464
465 line_ptr = CompBuffer;
466 line_end = comp_ptr;
467 break;
468
469 case 2 :
470 /*
471 * Do TIFF pack-bits encoding...
472 */
473
474 line_ptr = line;
475 line_end = line + length;
476 comp_ptr = CompBuffer;
477
478 while (line_ptr < line_end)
479 {
480 if ((line_ptr + 1) >= line_end)
481 {
482 /*
483 * Single byte on the end...
484 */
485
486 *comp_ptr++ = 0x00;
487 *comp_ptr++ = *line_ptr++;
488 }
489 else if (line_ptr[0] == line_ptr[1])
490 {
491 /*
492 * Repeated sequence...
493 */
494
495 line_ptr ++;
496 count = 2;
497
498 while (line_ptr < (line_end - 1) &&
499 line_ptr[0] == line_ptr[1] &&
500 count < 127)
501 {
502 line_ptr ++;
503 count ++;
504 }
505
506 *comp_ptr++ = (unsigned char)(257 - count);
507 *comp_ptr++ = *line_ptr++;
508 }
509 else
510 {
511 /*
512 * Non-repeated sequence...
513 */
514
515 start = line_ptr;
516 line_ptr ++;
517 count = 1;
518
519 while (line_ptr < (line_end - 1) &&
520 line_ptr[0] != line_ptr[1] &&
521 count < 127)
522 {
523 line_ptr ++;
524 count ++;
525 }
526
527 *comp_ptr++ = (unsigned char)(count - 1);
528
529 memcpy(comp_ptr, start, count);
530 comp_ptr += count;
531 }
532 }
533
534 line_ptr = CompBuffer;
535 line_end = comp_ptr;
536 break;
537 }
538
539 /*
540 * Set the length of the data and write a raster plane...
541 */
542
543 printf("\033*b%d%c", (int)(line_end - line_ptr), plane);
544 fwrite(line_ptr, (size_t)(line_end - line_ptr), 1, stdout);
545 }
546
547
548 /*
549 * 'OutputLine()' - Output a line of graphics.
550 */
551
552 void
OutputLine(cups_page_header2_t * header)553 OutputLine(cups_page_header2_t *header) /* I - Page header */
554 {
555 unsigned plane, /* Current plane */
556 bytes, /* Bytes to write */
557 count; /* Bytes to convert */
558 unsigned char bit, /* Current plane data */
559 bit0, /* Current low bit data */
560 bit1, /* Current high bit data */
561 *plane_ptr, /* Pointer into Planes */
562 *bit_ptr; /* Pointer into BitBuffer */
563
564
565 /*
566 * Output whitespace as needed...
567 */
568
569 if (Feed > 0)
570 {
571 printf("\033*b%dY", Feed);
572 Feed = 0;
573 }
574
575 /*
576 * Write bitmap data as needed...
577 */
578
579 bytes = (header->cupsWidth + 7) / 8;
580
581 for (plane = 0; plane < NumPlanes; plane ++)
582 if (ColorBits == 1)
583 {
584 /*
585 * Send bits as-is...
586 */
587
588 CompressData(Planes[plane], bytes, plane < (NumPlanes - 1) ? 'V' : 'W',
589 header->cupsCompression);
590 }
591 else
592 {
593 /*
594 * Separate low and high bit data into separate buffers.
595 */
596
597 for (count = header->cupsBytesPerLine / NumPlanes,
598 plane_ptr = Planes[plane], bit_ptr = BitBuffer;
599 count > 0;
600 count -= 2, plane_ptr += 2, bit_ptr ++)
601 {
602 bit = plane_ptr[0];
603
604 bit0 = (unsigned char)(((bit & 64) << 1) | ((bit & 16) << 2) | ((bit & 4) << 3) | ((bit & 1) << 4));
605 bit1 = (unsigned char)((bit & 128) | ((bit & 32) << 1) | ((bit & 8) << 2) | ((bit & 2) << 3));
606
607 if (count > 1)
608 {
609 bit = plane_ptr[1];
610
611 bit0 |= (unsigned char)((bit & 1) | ((bit & 4) >> 1) | ((bit & 16) >> 2) | ((bit & 64) >> 3));
612 bit1 |= (unsigned char)(((bit & 2) >> 1) | ((bit & 8) >> 2) | ((bit & 32) >> 3) | ((bit & 128) >> 4));
613 }
614
615 bit_ptr[0] = bit0;
616 bit_ptr[bytes] = bit1;
617 }
618
619 /*
620 * Send low and high bits...
621 */
622
623 CompressData(BitBuffer, bytes, 'V', header->cupsCompression);
624 CompressData(BitBuffer + bytes, bytes, plane < (NumPlanes - 1) ? 'V' : 'W',
625 header->cupsCompression);
626 }
627
628 fflush(stdout);
629 }
630
631
632 /*
633 * 'main()' - Main entry and processing of driver.
634 */
635
636 int /* O - Exit status */
main(int argc,char * argv[])637 main(int argc, /* I - Number of command-line arguments */
638 char *argv[]) /* I - Command-line arguments */
639 {
640 int fd; /* File descriptor */
641 cups_raster_t *ras; /* Raster stream for printing */
642 cups_page_header2_t header; /* Page header from file */
643 unsigned y; /* Current line */
644 ppd_file_t *ppd; /* PPD file */
645 #if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET)
646 struct sigaction action; /* Actions for POSIX signals */
647 #endif /* HAVE_SIGACTION && !HAVE_SIGSET */
648
649
650 /*
651 * Make sure status messages are not buffered...
652 */
653
654 setbuf(stderr, NULL);
655
656 /*
657 * Check command-line...
658 */
659
660 if (argc < 6 || argc > 7)
661 {
662 /*
663 * We don't have the correct number of arguments; write an error message
664 * and return.
665 */
666
667 _cupsLangPrintFilter(stderr, "ERROR",
668 _("%s job-id user title copies options [file]"),
669 "rastertohp");
670 return (1);
671 }
672
673 /*
674 * Open the page stream...
675 */
676
677 if (argc == 7)
678 {
679 if ((fd = open(argv[6], O_RDONLY)) == -1)
680 {
681 _cupsLangPrintError("ERROR", _("Unable to open raster file"));
682 sleep(1);
683 return (1);
684 }
685 }
686 else
687 fd = 0;
688
689 ras = cupsRasterOpen(fd, CUPS_RASTER_READ);
690
691 /*
692 * Register a signal handler to eject the current page if the
693 * job is cancelled.
694 */
695
696 Canceled = 0;
697
698 #ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */
699 sigset(SIGTERM, CancelJob);
700 #elif defined(HAVE_SIGACTION)
701 memset(&action, 0, sizeof(action));
702
703 sigemptyset(&action.sa_mask);
704 action.sa_handler = CancelJob;
705 sigaction(SIGTERM, &action, NULL);
706 #else
707 signal(SIGTERM, CancelJob);
708 #endif /* HAVE_SIGSET */
709
710 /*
711 * Initialize the print device...
712 */
713
714 ppd = ppdOpenFile(getenv("PPD"));
715 if (!ppd)
716 {
717 ppd_status_t status; /* PPD error */
718 int linenum; /* Line number */
719
720 _cupsLangPrintFilter(stderr, "ERROR",
721 _("The PPD file could not be opened."));
722
723 status = ppdLastError(&linenum);
724
725 fprintf(stderr, "DEBUG: %s on line %d.\n", ppdErrorString(status), linenum);
726
727 return (1);
728 }
729
730 Setup();
731
732 /*
733 * Process pages as needed...
734 */
735
736 Page = 0;
737
738 while (cupsRasterReadHeader2(ras, &header))
739 {
740 /*
741 * Write a status message with the page number and number of copies.
742 */
743
744 if (Canceled)
745 break;
746
747 Page ++;
748
749 fprintf(stderr, "PAGE: %d %d\n", Page, header.NumCopies);
750 _cupsLangPrintFilter(stderr, "INFO", _("Starting page %d."), Page);
751
752 /*
753 * Start the page...
754 */
755
756 StartPage(ppd, &header);
757
758 /*
759 * Loop for each line on the page...
760 */
761
762 for (y = 0; y < header.cupsHeight; y ++)
763 {
764 /*
765 * Let the user know how far we have progressed...
766 */
767
768 if (Canceled)
769 break;
770
771 if ((y & 127) == 0)
772 {
773 _cupsLangPrintFilter(stderr, "INFO",
774 _("Printing page %d, %u%% complete."),
775 Page, 100 * y / header.cupsHeight);
776 fprintf(stderr, "ATTR: job-media-progress=%u\n",
777 100 * y / header.cupsHeight);
778 }
779
780 /*
781 * Read a line of graphics...
782 */
783
784 if (cupsRasterReadPixels(ras, Planes[0], header.cupsBytesPerLine) < 1)
785 break;
786
787 /*
788 * See if the line is blank; if not, write it to the printer...
789 */
790
791 if (Planes[0][0] ||
792 memcmp(Planes[0], Planes[0] + 1, header.cupsBytesPerLine - 1))
793 OutputLine(&header);
794 else
795 Feed ++;
796 }
797
798 /*
799 * Eject the page...
800 */
801
802 _cupsLangPrintFilter(stderr, "INFO", _("Finished page %d."), Page);
803
804 EndPage();
805
806 if (Canceled)
807 break;
808 }
809
810 /*
811 * Shutdown the printer...
812 */
813
814 Shutdown();
815
816 if (ppd)
817 ppdClose(ppd);
818
819 /*
820 * Close the raster stream...
821 */
822
823 cupsRasterClose(ras);
824 if (fd != 0)
825 close(fd);
826
827 /*
828 * If no pages were printed, send an error message...
829 */
830
831 if (Page == 0)
832 {
833 _cupsLangPrintFilter(stderr, "ERROR", _("No pages were found."));
834 return (1);
835 }
836 else
837 return (0);
838 }
839