implement Pdriver; include "sys.m"; sys: Sys; include "draw.m"; draw: Draw; Display, Font, Rect, Point, Image, Screen: import draw; include "bufio.m"; bufio: Bufio; Iobuf: import bufio; include "print.m"; Printer: import Print; include "scaler.m"; scaler: Scaler; K: con 0; C: con 1; M: con 2; Y: con 3; Clight: con 4; Mlight: con 5; HPTRUE: con 1; HPFALSE: con 0; TRUE: con 1; FALSE: con 0; # RGB pixel RGB: adt { r, g, b: byte; }; # KCMY pixel KCMY: adt { k, c, m, y: byte; }; DitherParms: adt { fNumPix: int; fInput: array of byte; fErr: array of int; fSymmetricFlag: int; fFEDRes: array of int; fRasterEvenOrOdd: int; fHifipe: int; fOutput1, fOutput2, fOutput3: array of byte; }; # magic and wondrous HP colour maps map1: array of KCMY; map2: array of KCMY; ABSOLUTE: con 1; RELATIVE: con 0; Compression := 1; DEBUG := 0; stderr: ref Sys->FD; outbuf: ref Iobuf; ESC: con 27; # Palettes for Simple_Color PALETTE_RGB: con 3; PALETTE_CMY: con -3; PALETTE_KCMY: con -4; PALETTE_K: con 1; # Initialization init(debug: int) { sys = load Sys Sys->PATH; stderr = sys->fildes(2); draw = load Draw Draw->PATH; bufio = load Bufio Bufio->PATH; scaler = load Scaler Scaler->PATH; if (scaler == nil) fatal("Failed to load Scaler module"); DEBUG = debug; } # Return printable area in pixels printable_pixels(p: ref Printer): (int, int) { HMARGIN: con 0.6; WMARGIN: con 0.3; winches := p.popt.paper.width_inches - 2.0*WMARGIN; hinches := p.popt.paper.height_inches - 2.0*HMARGIN; wres := real p.popt.mode.resx; hres := real p.popt.mode.resy; (x, y) := (int (winches*wres), int (hinches*hres)); if (p.popt.orientation == Print->PORTRAIT) return (x, y); return (y, x); } # Send image to printer MASK := array[] of {byte 1, byte 3, byte 15, byte 255, byte 255}; SHIFT := array[] of {7, 6, 4, 0}; GSFACTOR := array[] of {255.0, 255.0/3.0, 255.0/7.0, 1.0, 1.0}; lastp : ref Printer; Refint: adt { value: int; }; watchdog(cancel: chan of int, cancelled: ref Refint) { <- cancel; cancelled.value = 1; } sendimage(p: ref Printer, pfd: ref Sys->FD, display: ref Draw->Display, im: ref Draw->Image, width: int, lmargin: int, cancel: chan of int): int { grppid := sys->pctl(Sys->NEWPGRP, nil); cancelled := ref Refint(0); spawn watchdog(cancel, cancelled); outopen(pfd); dbg(sys->sprint("image depth=%d from %d,%d to %d,%d\n", im.depth, im.r.min.x, im.r.min.y, im.r.max.x, im.r.max.y)); if (p != lastp) { (map1, map2) = readmaps(p); lastp = p; } bpp := im.depth; linechan := chan of array of int; if (p.popt.orientation == Print->PORTRAIT) InputWidth := im.r.max.x-im.r.min.x; else InputWidth = im.r.max.y-im.r.min.y; AdjustedInputWidth := (InputWidth+7) - ((InputWidth+7) % 8); dbg(sys->sprint("bpp=%d, InputWidth=%d, AdjustedInputWidth=%d\n", bpp, InputWidth, AdjustedInputWidth)); if (p.popt.orientation == Print->PORTRAIT) spawn row_by_row(im, linechan, AdjustedInputWidth); else spawn rotate(im, linechan, AdjustedInputWidth); DesiredOutputWidth := AdjustedInputWidth; if (width > AdjustedInputWidth) DesiredOutputWidth = width; ScaledWidth := 8*((DesiredOutputWidth)/8); mode := p.popt.mode; Nplanes := 4; if (map2 != nil) Nplanes += 2; Contone := array[Nplanes] of array of byte; ColorDepth := array[Nplanes] of int; ColorDepth[K] = mode.blackdepth; for (col:=1; col<Nplanes; col++) ColorDepth[col] = mode.coldepth; OutputWidth := array[Nplanes] of int; fDitherParms := array[Nplanes] of DitherParms; ErrBuff := array[Nplanes] of array of int; ColorPlane := array[Nplanes] of array of array of array of byte; MixedRes := 0; BaseResX := mode.resx; BaseResY := mode.resy; ResBoost := BaseResX / BaseResY; ResolutionX := array[Nplanes] of int; ResolutionY := array[Nplanes] of int; ResolutionX[K] = mode.resx*mode.blackresmult; ResolutionY[K] = mode.resy*mode.blackresmult; for (col=1; col<Nplanes; col++) { ResolutionX[col] = mode.resx; ResolutionY[col] = mode.resy; } NumRows := array[Nplanes] of int; for (j:=0; j<Nplanes; j++) { if (ResolutionX[j] != ResolutionX[K]) MixedRes++; if (MixedRes) # means res(K) !+ res(C,M,Y) NumRows[j] = ResolutionX[j] / BaseResX; else NumRows[j]=1; OutputWidth[j]= ScaledWidth * NumRows[j] * ResBoost; PlaneSize:= OutputWidth[j]/8; Contone[j] = array[OutputWidth[j]] of byte; ColorPlane[j] = array[NumRows[j]] of array of array of byte; for (jj:=0; jj<NumRows[j]; jj++) { ColorPlane[j][jj] = array[ColorDepth[j]] of array of byte; for (jjj:=0; jjj<ColorDepth[j]; jjj++) { ColorPlane[j][jj][jjj] = array[PlaneSize] of byte; } } ErrBuff[j] = array[OutputWidth[j]+2] of {* => 0}; } pcl_startjob(p); if (p.popt.paper.hpcode != "") PCL_Page_Size(p.popt.paper.hpcode); PCL_Move_CAP_H_Units(lmargin*300/BaseResX, ABSOLUTE); PCL_Configure_Raster_Data4(BaseResX, BaseResY, ColorDepth); PCL_Source_Raster_Width(ScaledWidth); PCL_Compression_Method(Compression); PCL_Start_Raster(1); cmap1 := setup_color_map(display, map1, im.depth); if (map2 != nil) cmap2 := setup_color_map(display, map2, im.depth); numerator, denominator: int; if ((ScaledWidth % AdjustedInputWidth)==0) { numerator = ScaledWidth / AdjustedInputWidth; denominator = 1; } else { numerator = ScaledWidth; denominator = AdjustedInputWidth; } rs := scaler->init(DEBUG, AdjustedInputWidth, numerator, denominator); rasterno := 0; col_row: array of int; eof := 0; while (!eof) { col_row = <- linechan; if (col_row == nil) eof++; scaler->rasterin(rs, col_row); while ((scaled_col_row := scaler->rasterout(rs)) != nil) { rasterno++; fRasterOdd := rasterno & 1; kcmy_row := SimpleColorMatch(cmap1, scaled_col_row); if (DEBUG) { dbg("Scaled Raster line:"); for (q:=0; q<len scaled_col_row; q++) { (r, g, b) := display.cmap2rgb(scaled_col_row[q]); dbg(sys->sprint("%d rgb=(%d,%d,%d) kcmy=(%d,%d,%d,%d)\n", int scaled_col_row[q], r, g, b, int kcmy_row[q].k, int kcmy_row[q].c, int kcmy_row[q].m, int kcmy_row[q].y)); } dbg("\n"); } Contone_K := Contone[K]; Contone_C := Contone[C]; Contone_M := Contone[M]; Contone_Y := Contone[Y]; for (ii:=0; ii<len Contone[K]; ii++) { kcmy := kcmy_row[ii]; Contone_K[ii] = kcmy.k; Contone_C[ii] = kcmy.c; Contone_M[ii] = kcmy.m; Contone_Y[ii] = kcmy.y; } if (map2 != nil) { # For lighter inks kcmy_row_light := SimpleColorMatch(cmap2, scaled_col_row); Contone_Clight := Contone[Clight]; Contone_Mlight := Contone[Mlight]; for (ii=0; ii<len Contone[Clight]; ii++) { kcmy := kcmy_row_light[ii]; Contone_Clight[ii] = kcmy.c; Contone_Mlight[ii] = kcmy.m; } } for (i:=0; i< Nplanes; i++) { # Pixel multiply here!! fDitherParms[i].fNumPix = OutputWidth[i]; fDitherParms[i].fInput = Contone[i]; fDitherParms[i].fErr = ErrBuff[i]; # fDitherParms[i].fErr++; // serpentine (?) fDitherParms[i].fSymmetricFlag = 1; # if (i == K) # fDitherParms[i].fFEDResPtr = fBlackFEDResPtr; # else # fDitherParms[i].fFEDResPtr = fColorFEDResPtr; fDitherParms[i].fFEDRes = FEDarray; fDitherParms[i].fRasterEvenOrOdd = fRasterOdd; fDitherParms[i].fHifipe = ColorDepth[i] > 1; for (j=0; j < NumRows[i]; j++) { fDitherParms[i].fOutput1 = ColorPlane[i][j][0]; if (fDitherParms[i].fHifipe) fDitherParms[i].fOutput2 = ColorPlane[i][j][1]; # dbg(sys->sprint("Dither for Row %d ColorPlane[%d][%d]\n", rasterno, i, j)); Dither(fDitherParms[i]); } } FINALPLANE: con 3; # NfinalPlanes := 4; for (i=0; i<=FINALPLANE; i++) { cp_i := ColorPlane[i]; coldepth_i := ColorDepth[i]; finalrow := NumRows[i]-1; for (j=0; j<=finalrow; j++) { cp_i_j := cp_i[j]; for (k:=0; k<coldepth_i; k++) { if (i == FINALPLANE && j == finalrow && k == coldepth_i-1) PCL_Transfer_Raster_Row(cp_i_j[k]); else PCL_Transfer_Raster_Plane(cp_i_j[k]); if (cancelled.value) { PCL_Reset(); outclose(); killgrp(grppid); return -1; } } } } } } PCL_End_Raster(); PCL_Reset(); outclose(); killgrp(grppid); if (cancelled.value) return -1; #sys->print("dlen %d, clen %d overruns %d\n", dlen, clen, overruns); return 0; } # Send text to printer sendtextfd(p: ref Print->Printer, pfd, tfd: ref Sys->FD, pointsize: real, proportional: int, wrap: int): int { outopen(pfd); pcl_startjob(p); if (wrap) PCL_End_of_Line_Wrap(0); LATIN1: con "0N"; PCL_Font_Symbol_Set(LATIN1); if (proportional) PCL_Font_Spacing(1); if (pointsize > 0.0) { PCL_Font_Height(pointsize); pitch := 10.0*12.0/pointsize; PCL_Font_Pitch(pitch); spacing := int (6.0*12.0/pointsize); PCL_Line_Spacing(spacing); dbg(sys->sprint("Text: pointsize %f pitch %f spacing %d\n", pointsize, pitch, spacing)); } PCL_Line_Termination(3); inbuf := bufio->fopen(tfd, Bufio->OREAD); while ((line := inbuf.gets('\n')) != nil) { ob := array of byte line; outwrite(ob, len ob); } PCL_Reset(); outclose(); return 0; } # Common PCL start pcl_startjob(p: ref Printer) { PCL_Reset(); if (p.popt.duplex) { esc("%-12345X@PJL DEFAULT DUPLEX=ON\n"); esc("%-12345X"); } if (p.popt.paper.hpcode != "") PCL_Page_Size(p.popt.paper.hpcode); PCL_Orientation(p.popt.orientation); PCL_Duplex(p.popt.duplex); } # Spawned to return sequence of rotated image rows rotate(im: ref Draw->Image, linechan: chan of array of int, adjwidth: int) { xmin := im.r.min.x; xmax := im.r.max.x; InputWidth := xmax - xmin; rawchan := chan of array of int; spawn row_by_row(im, rawchan, InputWidth); r_image := array[InputWidth] of {* => array [adjwidth] of {* => 0}}; r_row := 0; while ((col_row := <- rawchan) != nil) { endy := len col_row - 1; for (i:=0; i<len col_row; i++) r_image[endy - i][r_row] = col_row[i]; r_row++; } for (i:=0; i<len r_image; i++) linechan <-= r_image[i]; linechan <-= nil; } # Spawned to return sequence of image rows row_by_row(im: ref Draw->Image, linechan: chan of array of int, adjwidth: int) { xmin := im.r.min.x; ymin := im.r.min.y; xmax := im.r.max.x; ymax := im.r.max.y; InputWidth := xmax - xmin; bpp := im.depth; ld := ldepth(im.depth); bytesperline := (InputWidth*bpp+7)/8; rdata := array[bytesperline+10] of byte; pad0 := array [7] of { * => 0}; for (y:=ymin; y<ymax; y++) { col_row := array[adjwidth] of int; rect := Rect((xmin, y), (xmax, y+1)); np := im.readpixels(rect, rdata); if (np < 0) fatal("Error reading image\n"); dbg(sys->sprint("Input Raster line %d: np=%d\n ", y, np)); ind := 0; mask := MASK[ld]; shift := SHIFT[ld]; col_row[adjwidth-7:] = pad0; # Pad to adjusted width with white data := rdata[ind]; for (q:=0; q<InputWidth; q++) { col := int ((data >> shift) & mask); shift -= bpp; if (shift < 0) { shift = SHIFT[ld]; ind++; data = rdata[ind]; } col_row[q] = col; } linechan <-= col_row; } linechan <-= nil; } # PCL output routines PCL_Reset() { esc("E"); } PCL_Orientation(value: int) { esc(sys->sprint("&l%dO", value)); } PCL_Duplex(value: int) { esc(sys->sprint("&l%dS", value)); } PCL_Left_Margin(value: int) { esc(sys->sprint("&a%dL", value)); } PCL_Page_Size(value: string) { esc(sys->sprint("&l%sA", value)); } PCL_End_of_Line_Wrap(value: int) { esc(sys->sprint("&s%dC", value)); } PCL_Line_Termination(value: int) { esc(sys->sprint("&k%dG", value)); } PCL_Font_Symbol_Set(value: string) { esc(sys->sprint("(%s", value)); } PCL_Font_Pitch(value: real) { esc(sys->sprint("(s%2.2fH", value)); } PCL_Font_Spacing(value: int) { esc(sys->sprint("(s%dP", value)); } PCL_Font_Height(value: real) { esc(sys->sprint("(s%2.2fV", value)); } PCL_Line_Spacing(value: int) { esc(sys->sprint("&l%dD", value)); } PCL_Start_Raster(current: int) { flag := 0; if (current) flag = 1; esc(sys->sprint("*r%dA", flag)); } PCL_End_Raster() { esc("*rC"); } PCL_Raster_Resolution(ppi: int) { esc(sys->sprint("*t%dR", ppi)); } PCL_Source_Raster_Width(pixels: int) { esc(sys->sprint("*r%dS", pixels)); } PCL_Simple_Color(palette: int) { esc(sys->sprint("*r%dU", palette)); } PCL_Compression_Method(ctype: int) { esc(sys->sprint("*b%dM", ctype)); } PCL_Move_CAP_V_Rows(pos: int, absolute: int) { plus := ""; if (!absolute && pos > 0) plus = "+"; esc(sys->sprint("&a%s%dR", plus, pos)); } PCL_Move_CAP_H_Cols(pos: int, absolute: int) { plus := ""; if (!absolute && pos > 0) plus = "+"; esc(sys->sprint("&a%s%dC", plus, pos)); } # These Units are 1/300 of an inch. PCL_Move_CAP_H_Units(pos: int, absolute: int) { plus := ""; if (!absolute && pos > 0) plus = "+"; esc(sys->sprint("*p%s%dX", plus, pos)); } PCL_Move_CAP_V_Units(pos: int, absolute: int) { plus := ""; if (!absolute && pos > 0) plus = "+"; esc(sys->sprint("*p%s%dY", plus, pos)); } PCL_Configure_Raster_Data4(hres, vres: int, ColorDepth: array of int) { ncomponents := 4; msg := array[ncomponents*6 + 2] of byte; i := 0; msg[i++] = byte 2; # Format msg[i++] = byte ncomponents; # KCMY for (c:=0; c<ncomponents; c++) { msg[i++] = byte (hres/256); msg[i++] = byte (hres%256); msg[i++] = byte (vres/256); msg[i++] = byte (vres%256); depth := 1 << ColorDepth[c]; msg[i++] = byte (depth/256); msg[i++] = byte (depth%256); } if (DEBUG) { dbg("CRD: "); for (ii:=0; ii<len msg; ii++) dbg(sys->sprint("%d(%x) ", int msg[ii], int msg[ii])); dbg("\n"); } esc(sys->sprint("*g%dW", len msg)); outwrite(msg, len msg); } dlen := 0; clen := 0; overruns := 0; PCL_Transfer_Raster_Plane(data: array of byte) { if (DEBUG) { dbg("Transfer_Raster_Plane:"); for (i:=0; i<len data; i++) dbg(sys->sprint(" %x", int data[i])); dbg("\n"); } if (Compression) { d := len data; dlen += d; data = compress(data); c := len data; clen += c; if (c > d) overruns += c-d; if (DEBUG) { dbg("Compressed Transfer_Raster_Plane:"); for (i:=0; i<len data; i++) dbg(sys->sprint(" %x", int data[i])); dbg("\n"); } } esc(sys->sprint("*b%dV", len data)); outwrite(data, len data); } PCL_Transfer_Raster_Row(data: array of byte) { if (DEBUG) { dbg("Transfer_Raster_Row:"); for (i:=0; i<len data; i++) dbg(sys->sprint(" %x", int data[i])); dbg("\n"); } if (Compression) { data = compress(data); if (DEBUG) { dbg("Compressed Transfer_Raster_Row:"); for (i:=0; i<len data; i++) dbg(sys->sprint(" %x", int data[i])); dbg("\n"); } } esc(sys->sprint("*b%dW", len data)); outwrite(data, len data); } outopen(fd: ref Sys->FD) { outbuf = bufio->fopen(fd, Bufio->OWRITE); if (outbuf == nil) sys->fprint(stderr, "Failed to open output fd: %r\n"); } outclose() { outbuf.close(); } # Write to output using buffered io outwrite(data: array of byte, length: int) { outbuf.write(data, length); } # Send escape code to printer esc(s: string) { os := sys->sprint("%c%s", ESC, s); ob := array of byte os; outwrite(ob, len ob); } # Read all the maps readmaps(p: ref Printer): (array of KCMY, array of KCMY) { mapfile := p.ptype.hpmapfile; mapf1 := Pdriver->DATAPREFIX + mapfile + ".map"; m1 := read_map(mapf1); if (m1 == nil) fatal("Failed to read map file"); mapf2 := Pdriver->DATAPREFIX + mapfile + "_2.map"; m2 := read_map(mapf2); return (m1, m2); } # Read a map file read_map(mapfile: string) : array of KCMY { mf := bufio->open(mapfile, bufio->OREAD); if (mf == nil) return nil; CUBESIZE: con 9*9*9; marray := array[CUBESIZE] of KCMY; i := 0; while (i <CUBESIZE && (lstr := bufio->mf.gets('\n')) != nil) { (n, toks) := sys->tokenize(lstr, " \t"); if (n >= 4) { marray[i].k = byte int hd toks; toks = tl toks; marray[i].c = byte int hd toks; toks = tl toks; marray[i].m = byte int hd toks; toks = tl toks; marray[i].y = byte int hd toks; i++; } } return marray; } # Big interpolation routine # static data prev := RGB (byte 255, byte 255, byte 255); result: KCMY; offset := array[] of { 0, 1, 9, 10, 81, 82, 90, 91 }; Interpolate(map: array of KCMY, start: int, rgb: RGB, firstpixel: int): KCMY { cyan := array[8] of int; magenta := array[8] of int; yellow := array[8] of int; black := array[8] of int; if (firstpixel || prev.r != rgb.r || prev.g != rgb.g || prev.b != rgb.b) { prev = rgb; for (j:=0; j<8; j++) { ioff := start+offset[j]; cyan[j] = int map[ioff].c; magenta[j] = int map[ioff].m; yellow[j] = int map[ioff].y; black[j] = int map[ioff].k; } diff_red := int rgb.r & 16r1f; diff_green := int rgb.g & 16r1f; diff_blue := int rgb.b & 16r1f; result.c = byte (((cyan[0] + ( ( (cyan[4] - cyan[0] ) * diff_red) >> 5)) + ( ( ((cyan[2] + ( ( (cyan[6] - cyan[2] ) * diff_red) >> 5)) -(cyan[0] + ( ( (cyan[4] - cyan[0] ) * diff_red) >> 5)) ) * diff_green) >> 5)) + ( ( (((cyan[1] + ( ( (cyan[5] - cyan[1] ) * diff_red) >> 5)) + ( ( ((cyan[3] + ( ( (cyan[7] - cyan[3] ) * diff_red) >> 5)) -(cyan[1] + ( ( (cyan[5] - cyan[1] ) * diff_red) >> 5)) ) * diff_green) >> 5)) -((cyan[0] + ( ( (cyan[4] - cyan[0] ) * diff_red) >> 5)) + ( ( ((cyan[2] + ( ( (cyan[6] - cyan[2] ) * diff_red) >> 5)) -(cyan[0] + ( ( (cyan[4] - cyan[0] ) * diff_red) >> 5)) ) * diff_green) >> 5)) ) * diff_blue) >> 5)); result.m = byte (((magenta[0] + ( ( (magenta[4] - magenta[0] ) * diff_red) >> 5)) + ( ( ((magenta[2] + ( ( (magenta[6] - magenta[2] ) * diff_red) >> 5)) -(magenta[0] + ( ( (magenta[4] - magenta[0] ) * diff_red) >> 5)) ) * diff_green) >> 5)) + ( ( (((magenta[1] + ( ( (magenta[5] - magenta[1] ) * diff_red) >> 5)) + ( ( ((magenta[3] + ( ( (magenta[7] - magenta[3] ) * diff_red) >> 5)) -(magenta[1] + ( ( (magenta[5] - magenta[1] ) * diff_red) >> 5)) ) * diff_green) >> 5)) -((magenta[0] + ( ( (magenta[4] - magenta[0] ) * diff_red) >> 5)) + ( ( ((magenta[2] + ( ( (magenta[6] - magenta[2] ) * diff_red) >> 5)) -(magenta[0] + ( ( (magenta[4] - magenta[0] ) * diff_red) >> 5)) ) * diff_green) >> 5)) ) * diff_blue) >> 5)); result.y = byte (((yellow[0] + ( ( (yellow[4] - yellow[0] ) * diff_red) >> 5)) + ( ( ((yellow[2] + ( ( (yellow[6] - yellow[2] ) * diff_red) >> 5)) -(yellow[0] + ( ( (yellow[4] - yellow[0] ) * diff_red) >> 5)) ) * diff_green) >> 5)) + ( ( (((yellow[1] + ( ( (yellow[5] - yellow[1] ) * diff_red) >> 5)) + ( ( ((yellow[3] + ( ( (yellow[7] - yellow[3] ) * diff_red) >> 5)) -(yellow[1] + ( ( (yellow[5] - yellow[1] ) * diff_red) >> 5)) ) * diff_green) >> 5)) -((yellow[0] + ( ( (yellow[4] - yellow[0] ) * diff_red) >> 5)) + ( ( ((yellow[2] + ( ( (yellow[6] - yellow[2] ) * diff_red) >> 5)) -(yellow[0] + ( ( (yellow[4] - yellow[0] ) * diff_red) >> 5)) ) * diff_green) >> 5)) ) * diff_blue) >> 5)); result.k = byte (((black[0] + ( ( (black[4] - black[0] ) * diff_red) >> 5)) + ( ( ((black[2] + ( ( (black[6] - black[2] ) * diff_red) >> 5)) -(black[0] + ( ( (black[4] - black[0] ) * diff_red) >> 5)) ) * diff_green) >> 5)) + ( ( (((black[1] + ( ( (black[5] - black[1] ) * diff_red) >> 5)) + ( ( ((black[3] + ( ( (black[7] - black[3] ) * diff_red) >> 5)) -(black[1] + ( ( (black[5] - black[1] ) * diff_red) >> 5)) ) * diff_green) >> 5)) -((black[0] + ( ( (black[4] - black[0] ) * diff_red) >> 5)) + ( ( ((black[2] + ( ( (black[6] - black[2] ) * diff_red) >> 5)) -(black[0] + ( ( (black[4] - black[0] ) * diff_red) >> 5)) ) * diff_green) >> 5)) ) * diff_blue) >> 5)); } return result; } # Colour RGB to KCMY convertor ColorMatch(map: array of KCMY, row: array of RGB): array of KCMY { kcmy := array[len row] of KCMY; first := 1; for (i:=0; i<len row; i++) { r := int row[i].r; g := int row[i].g; b := int row[i].b; start := ((r & 16re0) << 1) + ((r & 16re0) >> 1) + (r >> 5) + ((g & 16re0) >> 2) + (g >> 5) + (b >> 5); kcmy[i] = Interpolate(map, start, row[i], first); # dbg(sys->sprint("+++ for (%d,%d,%d) Interpolate returned (%d,%d,%d,%d)\n", r, g, b, int kcmy[i].k, int kcmy[i].c, int kcmy[i].m, int kcmy[i].y)); first = 0; } return kcmy; } # Simple version of above to lookup precalculated values SimpleColorMatch(cmap: array of KCMY, colrow: array of int): array of KCMY { ncolrow := len colrow; kcmy_row := array[ncolrow] of KCMY; for (i:=0; i<ncolrow; i++) kcmy_row[i] = cmap[colrow[i]]; return kcmy_row; } ldepth(d: int): int { if(d & (d-1) || d >= 16) return 4; for(i := 0; i < 3; i++) if(d <= (1<<i)) break; return i; } # Set up color map once and for all setup_color_map(display: ref Display, map: array of KCMY, depth: int): array of KCMY { gsfactor := GSFACTOR[ldepth(depth)]; bpp := depth; max := 1 << bpp; rgb_row := array[max] of RGB; for (i:=0; i<max; i++) { if (depth >= 8) { (r, g, b) := display.cmap2rgb(i); rgb_row[i] = RGB (byte r, byte g, byte b); } else { # BW or Greyscale grey := byte (255-int (real i * gsfactor)); rgb_row[i] = RGB (grey, grey, grey); } } kcmy_row := ColorMatch(map, rgb_row); return kcmy_row; } # Dithering tmpShortStore: int; diffusionErrorPtr := 1; # for serpentine?? errPtr: array of int; rasterByte1 := 0; rasterByte2 := 0; rand8 := array [8] of int; pad8 := array [8] of {* => 0}; Dither(ditherParms: DitherParms) { errPtr = ditherParms.fErr; numLoop := ditherParms.fNumPix; inputPtr := 0; fedResTbl := ditherParms.fFEDRes; symmetricFlag := ditherParms.fSymmetricFlag; doNext8Pixels : int; hifipe := ditherParms.fHifipe; outputPtr1 := 0; outputPtr2 := 0; diffusionErrorPtr = 1; fInput := ditherParms.fInput; if(ditherParms.fRasterEvenOrOdd) { tmpShortStore = errPtr[diffusionErrorPtr]; errPtr[diffusionErrorPtr] = 0; for (pixelCount := numLoop + 8; (pixelCount -= 8) > 0; ) { if (pixelCount > 16) { # if next 16 pixels are white, skip 8 # doNext8Pixels = Forward16PixelsNonWhite(fInput, inputPtr); doNext8Pixels = 0; lim := inputPtr + 16; for (i := inputPtr; i < lim; i++) { if (fInput[i] != byte 0) { doNext8Pixels = 1; break; } } } else { doNext8Pixels = 1; } if (doNext8Pixels) { FORWARD_FED8(fInput, inputPtr, fedResTbl); inputPtr += 8; # HPRand8(); # FORWARD_FED(rand8[0], 16r80, fInput[inputPtr++], fedResTbl); # FORWARD_FED(rand8[1], 16r40, fInput[inputPtr++], fedResTbl); # FORWARD_FED(rand8[2], 16r20, fInput[inputPtr++], fedResTbl); # FORWARD_FED(rand8[3], 16r10, fInput[inputPtr++], fedResTbl); # FORWARD_FED(rand8[4], 16r08, fInput[inputPtr++], fedResTbl); # FORWARD_FED(rand8[5], 16r04, fInput[inputPtr++], fedResTbl); # FORWARD_FED(rand8[6], 16r02, fInput[inputPtr++], fedResTbl); # FORWARD_FED(rand8[7], 16r01, fInput[inputPtr++], fedResTbl); ditherParms.fOutput1[outputPtr1++] = byte rasterByte1; rasterByte1 = 0; if (hifipe) { ditherParms.fOutput2[outputPtr2++] = byte rasterByte2; rasterByte2 = 0; } } else { # Do white space skipping inputPtr += 8; ditherParms.fOutput1[outputPtr1++] = byte 0; if (hifipe) { ditherParms.fOutput2[outputPtr2++] = byte 0; } errPtr[diffusionErrorPtr:] = pad8; diffusionErrorPtr += 8; rasterByte1 = 0; rasterByte2 = 0; tmpShortStore = 0; } } # for pixelCount } else { rasterByte1 = 0; rasterByte2 = 0; inputPtr += ( numLoop-1 ); outputPtr1 += ( numLoop/8 - 1 ); outputPtr2 += ( numLoop/8 - 1 ); diffusionErrorPtr += ( numLoop-1 ); tmpShortStore = errPtr[diffusionErrorPtr]; errPtr[diffusionErrorPtr] = 0; for (pixelCount := numLoop + 8; (pixelCount -= 8) > 0; ) { if (pixelCount > 16) { # if next 16 pixels are white, skip 8 # doNext8Pixels = Backward16PixelsNonWhite(fInput, inputPtr); doNext8Pixels = 0; lim := inputPtr - 16; for (i := inputPtr; i > lim; i--) { if (fInput[i] != byte 0) { doNext8Pixels = 1; break; } } } else { doNext8Pixels = HPTRUE; } if (doNext8Pixels) { BACKWARD_FED8(fInput, inputPtr, fedResTbl); inputPtr -= 8; # HPRand8(); # BACKWARD_FED(rand8[0], 16r01, fInput[inputPtr--], fedResTbl); # BACKWARD_FED(rand8[1], 16r02, fInput[inputPtr--], fedResTbl); # BACKWARD_FED(rand8[2], 16r04, fInput[inputPtr--], fedResTbl); # BACKWARD_FED(rand8[3], 16r08, fInput[inputPtr--], fedResTbl); # BACKWARD_FED(rand8[4], 16r10, fInput[inputPtr--], fedResTbl); # BACKWARD_FED(rand8[5], 16r20, fInput[inputPtr--], fedResTbl); # BACKWARD_FED(rand8[6], 16r40, fInput[inputPtr--], fedResTbl); # BACKWARD_FED(rand8[7], 16r80, fInput[inputPtr--], fedResTbl); ditherParms.fOutput1[outputPtr1-- ]= byte rasterByte1; rasterByte1 = 0; if (hifipe) { ditherParms.fOutput2[outputPtr2--] = byte rasterByte2; rasterByte2 = 0; } } else { # Do white space skipping inputPtr -= 8; ditherParms.fOutput1[outputPtr1--] = byte 0; if (hifipe) { ditherParms.fOutput2[outputPtr2--] = byte 0; } diffusionErrorPtr -= 8; errPtr[diffusionErrorPtr:] = pad8; rasterByte1 = 0; rasterByte2 = 0; tmpShortStore = 0; } } } } # Take a step back Backward16PixelsNonWhite(ba: array of byte, inputPtr: int): int { lim := inputPtr - 16; for (i := inputPtr; i > lim; i--) { if (ba[i] != byte 0) return TRUE; } return FALSE; } # Take a step forward Forward16PixelsNonWhite(ba: array of byte, inputPtr: int): int { lim := inputPtr + 16; for (i := inputPtr; i < lim; i++) { if (ba[i] != byte 0) return TRUE; } return FALSE; } FORWARD_FED8(input: array of byte, ix: int, fedResTbl: array of int) { HPRand8(); randix := 0; for (bitMask := 16r80; bitMask; bitMask >>= 1) { tone := int input[ix++]; fedResPtr := tone << 2; level := fedResTbl[fedResPtr]; if (tone != 0) { tone = ( tmpShortStore + int fedResTbl[fedResPtr+1] ); if (tone >= rand8[randix++]) { tone -= 255; level++; } case (level) { 0=> break; 1=> rasterByte1 |= bitMask; break; 2=> rasterByte2 |= bitMask; break; 3=> rasterByte2 |= bitMask; rasterByte1 |= bitMask; break; 4=> break; 5=> rasterByte1 |= bitMask; break; 6=> rasterByte2 |= bitMask; break; 7=> rasterByte2 |= bitMask; rasterByte1 |= bitMask; break; } } else { tone = tmpShortStore; } halftone := tone >> 1; errPtr[diffusionErrorPtr++] = halftone; tmpShortStore = errPtr[diffusionErrorPtr] + (tone - halftone); } } #FORWARD_FED(thresholdValue: int, bitMask: int, toneb: byte, fedResTbl : array of int) #{ # tone := int toneb; # fedResPtr := (tone << 2); # level := fedResTbl[fedResPtr]; # if (tone != 0) { # tone = ( tmpShortStore + int fedResTbl[fedResPtr+1] ); # if (tone >= thresholdValue) { # tone -= 255; # level++; # } # case (level) { # 0=> # break; # 1=> # rasterByte1 |= bitMask; # break; # 2=> # rasterByte2 |= bitMask; # break; # 3=> # rasterByte2 |= bitMask; rasterByte1 |= bitMask; # break; # 4=> # break; # 5=> # rasterByte1 |= bitMask; # break; # 6=> # rasterByte2 |= bitMask; # break; # 7=> # rasterByte2 |= bitMask; rasterByte1 |= bitMask; # break; # } # } else { # tone = tmpShortStore; # } # halftone := tone >> 1; # errPtr[diffusionErrorPtr++] = halftone; # tmpShortStore = errPtr[diffusionErrorPtr] + (tone - halftone); ## dbg(sys->sprint("FORWARD_FED: thresh %d bitMask %x toneb %d => rasterbytes %d,%d,%d\n", thresholdValue, bitMask, int toneb, rasterByte1, rasterByte2)); #} BACKWARD_FED8(input: array of byte, ix: int, fedResTbl: array of int) { HPRand8(); randix := 0; for (bitMask := 16r01; bitMask <16r100; bitMask <<= 1) { tone := int input[ix--]; fedResPtr := (tone << 2); level := fedResTbl[fedResPtr]; if (tone != 0) { tone = ( tmpShortStore + int fedResTbl[fedResPtr+1] ); if (tone >= rand8[randix++]) { tone -= 255; level++; } case (level) { 0=> break; 1=> rasterByte1 |= bitMask; break; 2=> rasterByte2 |= bitMask; break; 3=> rasterByte2 |= bitMask; rasterByte1 |= bitMask; break; 4=> break; 5=> rasterByte1 |= bitMask; break; 6=> rasterByte2 |= bitMask; break; 7=> rasterByte2 |= bitMask; rasterByte1 |= bitMask; break; } } else { tone = tmpShortStore; } halftone := tone >> 1; errPtr[diffusionErrorPtr--] = halftone; tmpShortStore = errPtr[diffusionErrorPtr] + (tone - halftone); } } #BACKWARD_FED(thresholdValue: int, bitMask: int, toneb: byte, fedResTbl : array of int) #{ # tone := int toneb; # fedResPtr := (tone << 2); # level := fedResTbl[fedResPtr]; # if (tone != 0) { # tone = ( tmpShortStore + int fedResTbl[fedResPtr+1] ); # if (tone >= thresholdValue) { # tone -= 255; # level++; # } # case (level) { # 0=> # break; # 1=> # rasterByte1 |= bitMask; # break; # 2=> # rasterByte2 |= bitMask; # break; # 3=> # rasterByte2 |= bitMask; rasterByte1 |= bitMask; # break; # 4=> # break; # 5=> # rasterByte1 |= bitMask; # break; # 6=> # rasterByte2 |= bitMask; # break; # 7=> # rasterByte2 |= bitMask; rasterByte1 |= bitMask; # break; # } # } else { # tone = tmpShortStore; # } # halftone := tone >> 1; # errPtr[diffusionErrorPtr--] = halftone; # tmpShortStore = errPtr[diffusionErrorPtr] + (tone - halftone); ## dbg(sys->sprint("BACWARD_FED: thresh %d bitMask %x toneb %d => rasterbytes %d,%d,%d\n", thresholdValue, bitMask, int toneb, rasterByte1, rasterByte2)); #} # Pixel replication pixrep(in: array of RGB): array of RGB { out := array[2*len in] of RGB; for (i:=0; i<len in; i++) { out[i*2] = in[i]; out[i*2+1] = in[i]; } return out; } # Random numbers IM: con 139968; IA: con 3877; IC: con 29573; last := 42; # Use a really simple and quick random number generator HPRand(): int { return (74 * (last = (last* IA + IC) % IM) / IM ) + 5; } HPRand8() { for (i:= 0; i < 8; i++) rand8[i] = (74 * (last = (last* IA + IC) % IM) / IM ) + 5; } # Compression compress(rawdata: array of byte): array of byte { nraw := len rawdata; comp := array [2*nraw] of byte; # worst case ncomp := 0; for (i:=0; i<nraw;) { rpt := 0; val := rawdata[i++]; while (i<nraw && rpt < 255 && rawdata[i] == val) { rpt++; i++; } comp[ncomp++] = byte rpt; comp[ncomp++] = val; } return comp[0:ncomp]; } # Print error message and exit fatal(s: string) { sys->fprint(stderr, "%s\n", s); exit; } killgrp(pid: int) { sys->fprint(sys->open("/prog/" + string pid +"/ctl", Sys->OWRITE), "killgrp"); } dbg(s: string) { if (DEBUG) sys->fprint(stderr, "%s", s); } # Uninteresting constants FEDarray := array[1024] of { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 2 , 0 , 0 , 0 , 3 , 0 , 0 , 0 , 4 , 0 , 0 , 0 , 5 , 0 , 0 , 0 , 6 , 0 , 0 , 0 , 7 , 0 , 0 , 0 , 8 , 0 , 0 , 0 , 9 , 0 , 0 , 0 , 10 , 0 , 0 , 0 , 11 , 0 , 0 , 0 , 12 , 0 , 0 , 0 , 13 , 0 , 0 , 0 , 14 , 0 , 0 , 0 , 15 , 0 , 0 , 0 , 16 , 0 , 0 , 0 , 17 , 0 , 0 , 0 , 18 , 0 , 0 , 0 , 19 , 0 , 0 , 0 , 20 , 0 , 0 , 0 , 21 , 0 , 0 , 0 , 22 , 0 , 0 , 0 , 23 , 0 , 0 , 0 , 24 , 0 , 0 , 0 , 25 , 0 , 0 , 0 , 26 , 0 , 0 , 0 , 27 , 0 , 0 , 0 , 28 , 0 , 0 , 0 , 29 , 0 , 0 , 0 , 30 , 0 , 0 , 0 , 31 , 0 , 0 , 0 , 32 , 0 , 0 , 0 , 33 , 0 , 0 , 0 , 34 , 0 , 0 , 0 , 35 , 0 , 0 , 0 , 36 , 0 , 0 , 0 , 37 , 0 , 0 , 0 , 38 , 0 , 0 , 0 , 39 , 0 , 0 , 0 , 40 , 0 , 0 , 0 , 41 , 0 , 0 , 0 , 42 , 0 , 0 , 0 , 43 , 0 , 0 , 0 , 44 , 0 , 0 , 0 , 45 , 0 , 0 , 0 , 46 , 0 , 0 , 0 , 47 , 0 , 0 , 0 , 48 , 0 , 0 , 0 , 49 , 0 , 0 , 0 , 50 , 0 , 0 , 0 , 51 , 0 , 0 , 0 , 52 , 0 , 0 , 0 , 53 , 0 , 0 , 0 , 54 , 0 , 0 , 0 , 55 , 0 , 0 , 0 , 56 , 0 , 0 , 0 , 57 , 0 , 0 , 0 , 58 , 0 , 0 , 0 , 59 , 0 , 0 , 0 , 60 , 0 , 0 , 0 , 61 , 0 , 0 , 0 , 62 , 0 , 0 , 0 , 63 , 0 , 0 , 0 , 64 , 0 , 0 , 0 , 65 , 0 , 0 , 0 , 66 , 0 , 0 , 0 , 67 , 0 , 0 , 0 , 68 , 0 , 0 , 0 , 69 , 0 , 0 , 0 , 70 , 0 , 0 , 0 , 71 , 0 , 0 , 0 , 72 , 0 , 0 , 0 , 73 , 0 , 0 , 0 , 74 , 0 , 0 , 0 , 75 , 0 , 0 , 0 , 76 , 0 , 0 , 0 , 77 , 0 , 0 , 0 , 78 , 0 , 0 , 0 , 79 , 0 , 0 , 0 , 80 , 0 , 0 , 0 , 81 , 0 , 0 , 0 , 82 , 0 , 0 , 0 , 83 , 0 , 0 , 0 , 84 , 0 , 0 , 0 , 85 , 0 , 0 , 0 , 86 , 0 , 0 , 0 , 87 , 0 , 0 , 0 , 88 , 0 , 0 , 0 , 89 , 0 , 0 , 0 , 90 , 0 , 0 , 0 , 91 , 0 , 0 , 0 , 92 , 0 , 0 , 0 , 93 , 0 , 0 , 0 , 94 , 0 , 0 , 0 , 95 , 0 , 0 , 0 , 96 , 0 , 0 , 0 , 97 , 0 , 0 , 0 , 98 , 0 , 0 , 0 , 99 , 0 , 0 , 0 , 100 , 0 , 0 , 0 , 101 , 0 , 0 , 0 , 102 , 0 , 0 , 0 , 103 , 0 , 0 , 0 , 104 , 0 , 0 , 0 , 105 , 0 , 0 , 0 , 106 , 0 , 0 , 0 , 107 , 0 , 0 , 0 , 108 , 0 , 0 , 0 , 109 , 0 , 0 , 0 , 110 , 0 , 0 , 0 , 111 , 0 , 0 , 0 , 112 , 0 , 0 , 0 , 113 , 0 , 0 , 0 , 114 , 0 , 0 , 0 , 115 , 0 , 0 , 0 , 116 , 0 , 0 , 0 , 117 , 0 , 0 , 0 , 118 , 0 , 0 , 0 , 119 , 0 , 0 , 0 , 120 , 0 , 0 , 0 , 121 , 0 , 0 , 0 , 122 , 0 , 0 , 0 , 123 , 0 , 0 , 0 , 124 , 0 , 0 , 0 , 125 , 0 , 0 , 0 , 126 , 0 , 0 , 0 , 127 , 0 , 0 , 0 , 128 , 0 , 0 , 0 , 129 , 0 , 0 , 0 , 130 , 0 , 0 , 0 , 131 , 0 , 0 , 0 , 132 , 0 , 0 , 0 , 133 , 0 , 0 , 0 , 134 , 0 , 0 , 0 , 135 , 0 , 0 , 0 , 136 , 0 , 0 , 0 , 137 , 0 , 0 , 0 , 138 , 0 , 0 , 0 , 139 , 0 , 0 , 0 , 140 , 0 , 0 , 0 , 141 , 0 , 0 , 0 , 142 , 0 , 0 , 0 , 143 , 0 , 0 , 0 , 144 , 0 , 0 , 0 , 145 , 0 , 0 , 0 , 146 , 0 , 0 , 0 , 147 , 0 , 0 , 0 , 148 , 0 , 0 , 0 , 149 , 0 , 0 , 0 , 150 , 0 , 0 , 0 , 151 , 0 , 0 , 0 , 152 , 0 , 0 , 0 , 153 , 0 , 0 , 0 , 154 , 0 , 0 , 0 , 155 , 0 , 0 , 0 , 156 , 0 , 0 , 0 , 157 , 0 , 0 , 0 , 158 , 0 , 0 , 0 , 159 , 0 , 0 , 0 , 160 , 0 , 0 , 0 , 161 , 0 , 0 , 0 , 162 , 0 , 0 , 0 , 163 , 0 , 0 , 0 , 164 , 0 , 0 , 0 , 165 , 0 , 0 , 0 , 166 , 0 , 0 , 0 , 167 , 0 , 0 , 0 , 168 , 0 , 0 , 0 , 169 , 0 , 0 , 0 , 170 , 0 , 0 , 0 , 171 , 0 , 0 , 0 , 172 , 0 , 0 , 0 , 173 , 0 , 0 , 0 , 174 , 0 , 0 , 0 , 175 , 0 , 0 , 0 , 176 , 0 , 0 , 0 , 177 , 0 , 0 , 0 , 178 , 0 , 0 , 0 , 179 , 0 , 0 , 0 , 180 , 0 , 0 , 0 , 181 , 0 , 0 , 0 , 182 , 0 , 0 , 0 , 183 , 0 , 0 , 0 , 184 , 0 , 0 , 0 , 185 , 0 , 0 , 0 , 186 , 0 , 0 , 0 , 187 , 0 , 0 , 0 , 188 , 0 , 0 , 0 , 189 , 0 , 0 , 0 , 190 , 0 , 0 , 0 , 191 , 0 , 0 , 0 , 192 , 0 , 0 , 0 , 193 , 0 , 0 , 0 , 194 , 0 , 0 , 0 , 195 , 0 , 0 , 0 , 196 , 0 , 0 , 0 , 197 , 0 , 0 , 0 , 198 , 0 , 0 , 0 , 199 , 0 , 0 , 0 , 200 , 0 , 0 , 0 , 201 , 0 , 0 , 0 , 202 , 0 , 0 , 0 , 203 , 0 , 0 , 0 , 204 , 0 , 0 , 0 , 205 , 0 , 0 , 0 , 206 , 0 , 0 , 0 , 207 , 0 , 0 , 0 , 208 , 0 , 0 , 0 , 209 , 0 , 0 , 0 , 210 , 0 , 0 , 0 , 211 , 0 , 0 , 0 , 212 , 0 , 0 , 0 , 213 , 0 , 0 , 0 , 214 , 0 , 0 , 0 , 215 , 0 , 0 , 0 , 216 , 0 , 0 , 0 , 217 , 0 , 0 , 0 , 218 , 0 , 0 , 0 , 219 , 0 , 0 , 0 , 220 , 0 , 0 , 0 , 221 , 0 , 0 , 0 , 222 , 0 , 0 , 0 , 223 , 0 , 0 , 0 , 224 , 0 , 0 , 0 , 225 , 0 , 0 , 0 , 226 , 0 , 0 , 0 , 227 , 0 , 0 , 0 , 228 , 0 , 0 , 0 , 229 , 0 , 0 , 0 , 230 , 0 , 0 , 0 , 231 , 0 , 0 , 0 , 232 , 0 , 0 , 0 , 233 , 0 , 0 , 0 , 234 , 0 , 0 , 0 , 235 , 0 , 0 , 0 , 236 , 0 , 0 , 0 , 237 , 0 , 0 , 0 , 238 , 0 , 0 , 0 , 239 , 0 , 0 , 0 , 240 , 0 , 0 , 0 , 241 , 0 , 0 , 0 , 242 , 0 , 0 , 0 , 243 , 0 , 0 , 0 , 244 , 0 , 0 , 0 , 245 , 0 , 0 , 0 , 246 , 0 , 0 , 0 , 247 , 0 , 0 , 0 , 248 , 0 , 0 , 0 , 249 , 0 , 0 , 0 , 250 , 0 , 0 , 0 , 251 , 0 , 0 , 0 , 252 , 0 , 0 , 0 , 253 , 0 , 0 , 0 , 254 , 0 , 0 , 0 , 254 , 0 , 0 };