ImagerGrayDeviceImpl.mesa
Copyright © 1984 Xerox Corporation. All rights reserved.
Doug Wyatt, November 3, 1984 12:11:21 pm PST
DIRECTORY
Basics USING [bitsPerWord, LongMult],
ImagerColor,
ImagerDevice USING [Class, ClassRep, Device, DeviceRep, HalftoneParameters, RunProc],
ImagerPixelArray,
ImagerPixelMap USING [Create, DeviceRectangle, Function, PixelMap],
ImagerPixelRow,
ImagerSampler USING [CreateSampler, ObtainInterpolatedSamples, Sampler],
ImagerTransformation USING [Cat, PreRotate, Transformation, Translate],
PrincOps USING [BBTableSpace, BitBltTable, BitBltTablePtr, zBNDCK, zINC],
PrincOpsUtils USING [AlignedBBTable, BITBLT],
Real USING [FixC, RoundC],
Vector2 USING [VEC];
ImagerGrayDeviceImpl: CEDAR PROGRAM
IMPORTS Basics, ImagerPixelArray, ImagerPixelMap, ImagerPixelRow, ImagerSampler, ImagerTransformation, PrincOpsUtils, Real
~ BEGIN
Device: TYPE ~ ImagerDevice.Device;
RunProc: TYPE ~ ImagerDevice.RunProc;
HalftoneParameters: TYPE ~ ImagerDevice.HalftoneParameters;
DeviceRectangle: TYPE ~ ImagerPixelMap.DeviceRectangle;
VEC: TYPE ~ Vector2.VEC;
Transformation: TYPE ~ ImagerTransformation.Transformation;
Color: TYPE ~ ImagerColor.Color;
ConstantColor: TYPE ~ ImagerColor.ConstantColor;
SampledColor: TYPE ~ ImagerColor.SampledColor;
SpecialColor: TYPE ~ ImagerColor.SpecialColor;
ColorOperator: TYPE ~ ImagerColor.ColorOperator;
ColorOperatorRep: TYPE ~ ImagerColor.ColorOperatorRep;
Case: TYPE ~ {nil, constant, sampled};
Data: TYPE ~ REF DataRep;
DataRep:
TYPE ~
RECORD[
frame: ImagerPixelMap.PixelMap, -- the frame buffer, 8 bits per pixel
pixelsPerInch: REAL, -- declared resolution
case: Case ← nil, -- what type of color
function: ImagerPixelMap.Function ← [null, null], -- bitblt function
gray: CARDINAL ← 0, -- bitblt gray value
pa: ImagerPixelArray.PixelArray ← NIL, -- pixel array from sampled color
multiplier: CARDINAL ← 1, lgScale: INTEGER ← 0, -- scale pa samples to 8 bits
paToDevice: Transformation ← NIL, -- transformation from pa coords to display
source: ImagerPixelMap.PixelMap ← [0, 0, 0, 0, 0, 0, NIL], -- source values from pixel array
paSampler: ImagerSampler.Sampler ← NIL, -- for sampling pixel array
paRow: ImagerPixelRow.PixelRow ← NIL -- buffer for pixels sampled from pa
];
class: ImagerDevice.Class ~
NEW[ImagerDevice.ClassRep ← [
type: $GrayDisplay,
SetColor: SetColor,
SetPriority: SetPriority,
SetHalftone: SetHalftone,
MaskRuns: MaskRuns
]];
defaultPixelsPerInch:
REAL ~ 42;
Create:
PUBLIC
PROC[frame: ImagerPixelMap.PixelMap,
pixelsPerInch: REAL ← defaultPixelsPerInch] RETURNS[Device] ~ {
data: Data ~ NEW[DataRep ← [frame: frame, pixelsPerInch: pixelsPerInch]];
surfaceToDevice: Transformation ~ ImagerTransformation.Translate[[frame.sSize, 0]];
surfaceToDevice.PreRotate[90];
data.paRow ← ImagerPixelRow.CreatePixelRow[sMin: 0, fMin: 0, fSize: frame.fSize];
RETURN[
NEW[ImagerDevice.DeviceRep ← [class: class,
clipBox: [sMin: 0, fMin: 0, sSize: frame.sSize, fSize: frame.fSize],
surfaceToDevice: surfaceToDevice,
surfaceUnitsPerInch: [pixelsPerInch, pixelsPerInch],
surfaceUnitsPerPixel: 1,
data: data]]];
};
Val: TYPE ~ ImagerPixelArray.Val;
Table: TYPE ~ REF TableRep;
TableRep: TYPE ~ RECORD[SEQUENCE size: NAT OF CARDINAL];
BuildTable:
PROC[max: Val, scale:
REAL, sWhite, sBlack:
REAL,
map: ImagerColor.SampleMap ← NIL] RETURNS[Table] ~ {
table: Table ~ NEW[TableRep[max+1]];
FOR s0: Val
IN[0..max]
DO
s: REAL ~ IF map=NIL THEN s0 ELSE map[s0];
v: REAL ~ MIN[MAX[(s-sBlack)/(sWhite-sBlack), 0], 1];
table[s0] ← Real.FixC[scale*v];
ENDLOOP;
RETURN[table];
};
SetColor:
PROC[device: Device, color: Color, viewToDevice: Transformation] ~ {
data: Data ~ NARROW[device.data];
WITH color
SELECT
FROM
color: ConstantColor => {
pixel: [0..256) ~ Real.RoundC[MIN[MAX[color.cie.Y, 0], 1]*255];
data.function ← [null, null];
data.gray ← pixel*256+pixel;
data.case ← constant;
};
color: SampledColor => {
pa: ImagerPixelArray.PixelArray ~ color.pa;
samples: ImagerPixelArray.Row ~ NEW[ImagerPixelArray.RowRep[pa.fSize]];
row: ImagerPixelRow.PixelRow ~ ImagerPixelRow.CreatePixelRow[
sMin: 0, fMin: 0, fSize: pa.fSize];
data.pa ← pa;
data.paToDevice ← ImagerTransformation.Cat[pa.m, color.um, viewToDevice];
data.source ← ImagerPixelMap.Create[lgBitsPerPixel: 3,
bounds: [sMin: 0, fMin: 0, sSize: pa.sSize, fSize: pa.fSize]];
WITH color.colorOperator
SELECT
FROM
op:
REF ColorOperatorRep.grayLinear => {
table: Table ← NIL;
IF pa.samplesPerPixel#1 THEN ERROR; -- samplesPerPixel must be 1
table ← BuildTable[max: pa.MaxSampleValue[0], scale: 255,
sWhite: op.sWhite, sBlack: op.sBlack, map: op.map];
FOR s:
NAT
IN[0..pa.sSize)
DO
ImagerPixelArray.GetRow[pa, samples, s, 0, 0];
FOR f: NAT IN[0..pa.fSize) DO row[f] ← table[samples[f]] ENDLOOP;
row.sOrigin ← s;
ImagerPixelRow.StorePixelRow[row, data.source];
ENDLOOP;
};
op:
REF ColorOperatorRep.separations => {
tables: ARRAY[0..4) OF Table ← ALL[NIL];
IF pa.samplesPerPixel#op.samplesPerPixel THEN ERROR; -- samplesPerPixel must match
FOR i:
NAT
IN[0..pa.samplesPerPixel)
DO
sep: ImagerColor.Separation ~ op[i];
tables[i] ← BuildTable[max: pa.MaxSampleValue[i], scale: 255*sep.cie.
Y,
sWhite: sep.sMax, sBlack: sep.sMin, map: sep.map];
ENDLOOP;
FOR s:
NAT
IN[0..pa.sSize)
DO
row.sOrigin ← s;
ImagerPixelRow.ClearPixelRow[row];
FOR i:
NAT
IN[0..pa.samplesPerPixel)
DO
table: Table ~ tables[i];
ImagerPixelArray.GetRow[pa, samples, s, 0, i];
FOR f: NAT IN[0..pa.fSize) DO row[f] ← row[f]+table[samples[f]] ENDLOOP;
ENDLOOP;
ImagerPixelRow.StorePixelRow[row, data.source];
ENDLOOP;
};
op: REF ColorOperatorRep.black => {
IF pa.samplesPerPixel=1 AND pa.MaxSampleValue[0]=1 THEN NULL
ELSE ERROR; -- must be 1 bit per pixel
};
op: REF ColorOperatorRep.gray => {
IF pa.samplesPerPixel=1 THEN NULL
ELSE ERROR; -- samplesPerPixel must be 1
IF op.map=NIL OR op.map.size>pa.MaxSampleValue[0] THEN NULL
ELSE ERROR; -- incompatible map size
SELECT pa.MaxSampleValue[0] FROM
1 => { data.multiplier ← 255; data.lgScale ← 0 };
255 => { data.multiplier ← 1; data.lgScale ← 0 };
ENDCASE => ERROR; -- unexpected maxSampleValue
};
op: REF ColorOperatorRep.map => {
IF pa.samplesPerPixel#1 THEN ERROR; -- samplesPerPixel must be 1
IF op.size>pa.MaxSampleValue[0] THEN NULL
ELSE ERROR; -- incompatible map size
};
ENDCASE => ERROR; -- unknown ColorOperator
data.paSampler ← ImagerSampler.CreateSampler[
m: data.paToDevice, x: 0, y: 0, sPixels: pa.sSize, fPixels: pa.fSize];
data.case ← sampled;
};
color: SpecialColor =>
SELECT color.ref
FROM
$XOR => {
data.function ← [xor, null];
data.gray ← 177777B;
data.case ← constant;
};
ENDCASE => ERROR; -- unknown special color
ENDCASE => ERROR; -- unknown color variant
};
SetPriority:
PROC[device: Device, priorityImportant:
BOOL] ~ {
};
SetHalftone:
PROC[device: Device, halftone: HalftoneParameters] ~ {
};
nullBitBltTable: PrincOps.BitBltTable ~ [
dst: [word: NIL, bit: 0], dstBpl: 0,
src: [word: NIL, bit: 0], srcDesc: [srcBpl[0]],
width: 0, height: 0, flags: []];
Check:
PROC[x:
INTEGER, max:
NAT]
RETURNS[
NAT] ~
TRUSTED MACHINE CODE { PrincOps.zINC; PrincOps.zBNDCK };
IF x IN[0..max] THEN RETURN[x] ELSE ERROR RuntimeError.BoundsFault
MaskRuns:
PROC[device: Device, runs:
PROC[RunProc]] ~ {
data: Data ~ NARROW[device.data];
frame: ImagerPixelMap.PixelMap ~ data.frame;
SELECT data.case
FROM
nil => ERROR; -- color not initialized
constant =>
TRUSTED {
bbspace: PrincOps.BBTableSpace;
bb: PrincOps.BitBltTablePtr ~ PrincOpsUtils.AlignedBBTable[@bbspace];
gray: LONG POINTER ~ @data.gray;
base: LONG POINTER ~ frame.refRep.pointer;
rast: CARDINAL ~ frame.refRep.rast;
run:
PROC[sMin, fMin:
INTEGER, fSize:
NAT] ~
TRUSTED {
f: NAT ~ Check[fMin, frame.fSize];
s: NAT ~ Check[sMin, frame.sSize];
bb.dst.word ← base+Basics.LongMult[s, rast]+f/2;
bb.dst.bit ← 8*(f MOD 2);
bb.width ← 8*Check[fSize, frame.fSize-f];
bb.height ← Check[1, frame.sSize-s];
PrincOpsUtils.BITBLT[bb];
};
bb^ ← nullBitBltTable;
bb.dstBpl ← rast*Basics.bitsPerWord;
bb.src.word ← gray;
bb.src.bit ← 0;
bb.srcDesc.gray ← [yOffset: 0, widthMinusOne: 0, heightMinusOne: 0];
bb.flags ← [direction: forward, disjoint:
TRUE, gray:
TRUE,
srcFunc: data.function.srcFunc, dstFunc: data.function.dstFunc];
runs[run];
};
sampled => {
paRow: ImagerPixelRow.PixelRow ~ data.paRow;
source: ImagerPixelMap.PixelMap ~ data.source;
run:
PROC[sMin, fMin:
INTEGER, fSize:
NAT] ~ {
paRow.sOrigin ← Check[sMin, frame.sSize];
paRow.fOrigin ← Check[fMin, frame.fSize];
paRow.fSize ← Check[fSize, frame.fSize-fMin];
ImagerSampler.ObtainInterpolatedSamples[sampler: data.paSampler,
pixelRow: paRow, source: source, multiplier: data.multiplier, lgScale: data.lgScale];
ImagerPixelRow.StorePixelRow[paRow, frame];
};
runs[run];
};
ENDCASE => ERROR; -- illegal case
};
END.