-- File CIFCIF.mesa
-- Output module for CIF 2.0
-- Written by Martin Newell, March 1981
-- Last updated: June 12, 1981 3:26 PM
DIRECTORY
AuxIntDefs: FROM "AuxIntDefs" USING [IExpand, IGetRootID, IGetFirstSymbol,
IGetNextSymbol, IStop],
AuxOutputDefs: FROM "AuxOutputDefs",
CIFDevicesDefs: FROM "CIFDevicesDefs" USING [DeviceDescriptor,
DeviceDescriptorRecord, RegisterDevice, MaxLENGTHLayerArray],
CIFUtilitiesDefs: FROM "CIFUtilitiesDefs" USING [Rectangle,
SetClipRectangle, DrawClipRectangle,
GetDisplayContext],
Graphics: FROM "Graphics" USING [DisplayContext, PopContext, PushContext,
Scale],
InlineDefs: FROM "InlineDefs" USING [LowHalf],
IntStorageDefs: FROM "IntStorageDefs" USING [ObjectName, NilObjectName],
IntTransDefs: FROM "IntTransDefs" USING [Transform, TransformRecord],
IODefs: FROM "IODefs" USING [CR, SP, NUL, WriteLine, WriteString,
WriteChar, GetOutputStream, SetOutputStream, WriteDecimal],
JaMFnsDefs: FROM "JaMFnsDefs" USING [PopString, GetJaMBreak, SetJaMBreak,
PopBoolean],
ParserTypeDefs: FROM "ParserTypeDefs" USING [Point, Path, RemovePoint],
Real: FROM "Real" USING [Fix, WriteReal],
StreamDefs: FROM "StreamDefs" USING [StreamHandle, NewByteStream, Write,
Append],
StringDefs: FROM "StringDefs" USING[AppendChar, AppendString,
AppendLongDecimal],
SystemDefs: FROM "SystemDefs" USING [AllocateHeapNode, FreeHeapNode];
CIFCIF: PROGRAM
IMPORTS AuxIntDefs, CIFDevicesDefs, CIFUtilitiesDefs, Graphics, InlineDefs, IODefs, JaMFnsDefs, ParserTypeDefs, Real, StreamDefs, StringDefs, SystemDefs
EXPORTS AuxOutputDefs =
BEGIN OPEN AuxIntDefs, AuxOutputDefs, CIFDevicesDefs, CIFUtilitiesDefs, Graphics, InlineDefs, IntStorageDefs, IntTransDefs, IODefs, JaMFnsDefs, ParserTypeDefs, Real, StreamDefs, StringDefs, SystemDefs;
-- CIF procedures
CIFDeviceRecord: DeviceDescriptorRecord ← [
next:NIL,
name:"cif",
deviceSelect: CIFSelect,
deviceDrawFrame: CIFDrawFrame,
deviceSetScale: CIFSetScale,
deviceSetClipRegion: CIFSetClipRegion,
deviceOutput: CIFOutput,
deviceLayer: CIFLayer,
deviceLoadLayer: CIFLoadLayer,
deviceRectangle: CIFRectangle,
deviceStartPoly: CIFStartPoly,
devicePolyVertex: CIFPolyVertex,
deviceEndPoly: CIFEndPoly,
deviceText: CIFText
];
--the following types are copied from CIFOutput
UserObjectType: TYPE = {User9, User94, other};
UserObject: TYPE = POINTER TO UserObjectRecord;
UserObjectRecord: TYPE = RECORD[
SELECT type: UserObjectType FROM
User9 => [--Symbol name
name: PACKED ARRAY [0..0) OF CHARACTER],
User94 => [--Named point
x: REAL,
y: REAL,
name: PACKED ARRAY [0..0) OF CHARACTER],
ENDCASE];
CIFSelect: PROCEDURE RETURNS[BOOLEAN] =
BEGIN
RETURN[TRUE];
END;
CIFDrawFrame: PROCEDURE =
BEGIN
DrawClipRectangle[];
END;
CIFSetScale: PROCEDURE [factor: REAL] =
BEGIN
dc: DisplayContext ← GetDisplayContext[];
PopContext[dc];
PushContext[dc];
Scale[dc, [factor,factor]];
END;
CIFSetClipRegion: PROCEDURE [rt: Rectangle] =
BEGIN
SetClipRectangle[rt];
END;
CIFOutput: PROCEDURE =
--expects <boolean filename> (boolean is true for ALL symbols, false for REFERENCED only)
BEGIN
symbol: ObjectName;
cifNum: CARDINAL;
fileName: STRING ← [100];
allSymbols: BOOLEAN;
PopString[fileName];
allSymbols ← PopBoolean[];
IF ~DotinName[fileName] THEN AppendString[fileName,".cif"];
Aborted ← FALSE;
InitDict[256];
IF allSymbols THEN --get handle on all symbols, even if not referenced
{FOR s:ObjectName ← IGetFirstSymbol[],IGetNextSymbol[]
UNTIL s=NilObjectName DO
{u: STRING ← [100];
[] ← Lookup[s];
};
ENDLOOP;
};
--provoke initial entries in dictionary
ScanningTopLevel ← TRUE;
IExpand[IGetRootID[]];
IF Aborted THEN
{Abort[];
RETURN;
};
ScanningTopLevel ← FALSE;
--output symbols
DoHeader[fileName];
[symbol,cifNum] ← GetNonProcessedEntry[];
UNTIL cifNum=0 DO
DoSymbolHeader[symbol];
IExpand[symbol];
IF Aborted THEN
{Abort[];
RETURN;
};
DoSymbolTrailer[symbol];
[symbol,cifNum] ← GetNonProcessedEntry[];
ENDLOOP;
--output mainline
CIFStream.put[CIFStream,CR];
IExpand[IGetRootID[]];
IF Aborted THEN
{Abort[];
RETURN;
};
DoTrailer[];
FreeDict[];
END;
Abort: PROCEDURE[] =
BEGIN
save: StreamHandle ← GetOutputStream[];
SetOutputStream[CIFStream];
WriteLine["***Aborted***"];
SetOutputStream[save];
WriteLine["***Aborted***"];
DoTrailer[];
FreeDict[];
END;
DoHeader: PROCEDURE[fileName: STRING] =
BEGIN
save: StreamHandle ← GetOutputStream[];
CIFStream ← NewByteStream[fileName, Write+Append];
SetOutputStream[CIFStream];
WriteString["(File: "]; WriteString[fileName];
WriteLine[". Created by CIFCIF);"];
SetOutputStream[save];
CurrentLayer ← 32000; --i.e. not set
END;
DoTrailer: PROCEDURE =
BEGIN
save: StreamHandle ← GetOutputStream[];
SetOutputStream[CIFStream];
WriteLine["End"];
SetOutputStream[save];
CIFStream.destroy[CIFStream];
CIFStream ← NIL;
END;
DoSymbolHeader: PROCEDURE [symbol: ObjectName] =
BEGIN
save: StreamHandle ← GetOutputStream[];
SetOutputStream[CIFStream];
WriteChar[CR];
WriteString["DS "]; WriteDecimal[Lookup[symbol]]; WriteLine[";"];
SaveLayer ← CurrentLayer;
CurrentLayer ← 32000;
SetOutputStream[save];
END;
DoSymbolTrailer: PROCEDURE[symbol: ObjectName] =
BEGIN
save: StreamHandle ← GetOutputStream[];
SetOutputStream[CIFStream];
WriteLine["DF;"];
SetOutputStream[save];
CurrentLayer ← SaveLayer;
END;
CIFLayer: PROCEDURE [layer: CARDINAL] =
BEGIN
END;
CIFLoadLayer: PROCEDURE[layer:CARDINAL, v0,v1,v2,v3: CARDINAL] =
BEGIN
END;
CIFRectangle: PROCEDURE [r: Rectangle] =
BEGIN
END;
CIFStartPoly: PROCEDURE [x,y: REAL] =
BEGIN
END;
CIFPolyVertex: PROCEDURE [x,y: REAL] =
BEGIN
END;
CIFEndPoly: PROCEDURE =
BEGIN
END;
CIFText: PROCEDURE[text: STRING, x,y: REAL] =
BEGIN
END;
--Procedures that export AuxOutputDefs--
AuxWire: PUBLIC PROCEDURE [layerName: CARDINAL, width: LONG CARDINAL, a: ParserTypeDefs.Path] =
BEGIN
save: StreamHandle ← GetOutputStream[];
gotone: BOOLEAN;
p: Point;
IF ScanningTopLevel THEN RETURN;
IF a.length=0 THEN RETURN;
SetOutputStream[CIFStream];
SetLayer[layerName];
WriteString["W "]; WriteLongDecimal[width];
[gotone,p] ← RemovePoint[a];
WHILE gotone DO
WriteChar[SP]; WriteLongDecimal[p.x];
WriteChar[SP]; WriteLongDecimal[p.y];
[gotone,p] ← RemovePoint[a];
ENDLOOP;
WriteLine[";"];
SetOutputStream[save];
END;
AuxFlash: PUBLIC PROCEDURE [layerName: CARDINAL, diameter: LONG CARDINAL, center: ParserTypeDefs.Point] =
BEGIN
save: StreamHandle ← GetOutputStream[];
IF ScanningTopLevel THEN RETURN;
SetOutputStream[CIFStream];
SetLayer[layerName];
WriteString["R "]; WriteLongDecimal[diameter];
WriteChar[SP]; WriteLongDecimal[center.x];
WriteChar[SP]; WriteLongDecimal[center.y];
WriteLine[";"];
SetOutputStream[save];
END;
AuxPolygon: PUBLIC PROCEDURE [layerName: CARDINAL, a: ParserTypeDefs.Path] =
BEGIN
save: StreamHandle ← GetOutputStream[];
gotone: BOOLEAN;
p: Point;
IF ScanningTopLevel THEN RETURN;
IF a.length=0 THEN RETURN;
SetOutputStream[CIFStream];
SetLayer[layerName];
WriteString["P"];
[gotone,p] ← RemovePoint[a];
WHILE gotone DO
WriteChar[SP]; WriteLongDecimal[p.x];
WriteChar[SP]; WriteLongDecimal[p.y];
[gotone,p] ← RemovePoint[a];
ENDLOOP;
WriteLine[";"];
SetOutputStream[save];
END;
AuxBox: PUBLIC PROCEDURE [layerName: CARDINAL, length, width: LONG CARDINAL,
center: ParserTypeDefs.Point, xRotation, yRotation: LONG INTEGER] =
BEGIN
save: StreamHandle ← GetOutputStream[];
IF GetJaMBreak[] THEN
{IStop[];
SetJaMBreak[FALSE];
Aborted ← TRUE;
RETURN;
};
IF ScanningTopLevel THEN RETURN;
SetOutputStream[CIFStream];
SetLayer[layerName];
WriteString["B "]; WriteLongDecimal[length];
WriteChar[SP]; WriteLongDecimal[width];
WriteChar[SP]; WriteLongDecimal[center.x];
WriteChar[SP]; WriteLongDecimal[center.y];
IF yRotation#0 OR xRotation<0 THEN
{WriteChar[SP]; WriteLongDecimal[xRotation];
WriteChar[SP]; WriteLongDecimal[yRotation];
};
WriteLine[";"];
SetOutputStream[save];
END;
AuxUserObject: PUBLIC PROCEDURE [layerName: CARDINAL,
size: CARDINAL, data: POINTER TO UNSPECIFIED] =
BEGIN
userObject: UserObject ← data;
save: StreamHandle ← GetOutputStream[];
IF ScanningTopLevel THEN RETURN;
WITH userObject SELECT FROM
User9 =>
{string: STRING ← [100];
i,stringlength: INTEGER;
SetOutputStream[CIFStream];
stringlength ← (size-SIZE[User9 UserObjectRecord])*2;
FOR i IN [0..stringlength) DO
AppendChar[string,IF name[i]=SP THEN ’* ELSE name[i]];
ENDLOOP;
IF stringlength>0 AND name[stringlength-1]=NUL THEN
string.length ← string.length - 1;
WriteString["9 "]; WriteString[string];
WriteLine[";"];
SetOutputStream[save];
};
User94 =>
{string: STRING ← [100];
i,stringlength: INTEGER;
SetOutputStream[CIFStream];
stringlength ← (size-SIZE[User94 UserObjectRecord])*2;
FOR i IN [0..stringlength) DO
AppendChar[string,IF name[i]=SP THEN ’* ELSE name[i]];
ENDLOOP;
IF stringlength>0 AND name[stringlength-1]=NUL THEN
string.length ← string.length - 1;
WriteString["94 "]; WriteString[string];
WriteChar[SP]; WriteFloat[x];
WriteChar[SP]; WriteFloat[y];
WriteLine[";"];
SetOutputStream[save];
};
ENDCASE => WriteLine["unknown user object"];
END;
AuxCall: PUBLIC PROCEDURE [symbol: ObjectName, cifNumber: LONG CARDINAL, transform: Transform] =
BEGIN
save: StreamHandle ← GetOutputStream[];
t: TransformRecord;
c,s: REAL;
Rx,Ry: LONG INTEGER;
cifNum: INTEGER ← Lookup[symbol]; --need to do this always
IF ScanningTopLevel THEN RETURN;
SetOutputStream[CIFStream];
WriteString["C "]; WriteLongDecimal[cifNum];
t ← transform↑;
--Check and output Mirror
IF (t.a11*t.a22-t.a21*t.a12)<0 THEN --determinant negative - need mirror
{t.a11 ← -t.a11;
t.a12 ← -t.a12;
WriteString[" MX"];
};
--check and output Rotate
c ← t.a11/t.a33;
s ← t.a12/t.a33;
IF s#0 OR c<0 THEN
{SELECT TRUE FROM
c=s=> Rx ← Ry ← 1;
c=0=> {Rx ← 0; Ry ← IF s<0 THEN -1 ELSE 1; };
s=0=> {Ry ← 0; Rx ← IF c<0 THEN -1 ELSE 1; };
ENDCASE =>
{f: REAL ← ABS[BigCIFInteger/c];
f ← MIN[f,ABS[BigCIFInteger/s]];
Rx ← Fix[c*f];
Ry ← Fix[s*f];
};
WriteString[" R "]; WriteLongDecimal[Rx];
WriteChar[SP]; WriteLongDecimal[Ry];
};
--Check and output translation
IF t.a31#0 OR t.a32#0 THEN
{WriteString[" T "]; WriteLongDecimal[Fix[t.a31/t.a33]];
WriteChar[SP]; WriteLongDecimal[Fix[t.a32/t.a33]];
};
WriteLine[";"];
SetOutputStream[save];
END;
BigCIFInteger: LONG INTEGER ← 20000000B; --2↑22
SetLayer: PUBLIC PROCEDURE [layerName: CARDINAL] =
BEGIN
IF CurrentLayer#layerName THEN
{save: StreamHandle ← GetOutputStream[];
SetOutputStream[CIFStream];
WriteString["L N"]; WriteChar[LayerChar[layerName]];
WriteString["; "];
SetOutputStream[save];
CurrentLayer ← layerName;
};
END;
--Dictionary
DictEntry: TYPE = POINTER TO DictEntryRecord;
DictEntryRecord: TYPE = RECORD [
next: DictEntry,--global list in order in which entries are made
ovflow: DictEntry,--overflow list from entry in hash table
key: ObjectName,--internal ID of symbol
value: CARDINAL--generated CIF symbol # for output
];
DictHead: DictEntry;
DictTail: DictEntry;
HTable: DESCRIPTOR FOR ARRAY OF DictEntry;
LastProcessedEntry: DictEntry; --points to last processed entry
Value: CARDINAL;
InitDict: PROCEDURE[hashlength: CARDINAL] =
BEGIN
i: CARDINAL;
DictHead ← NIL;
DictTail ← NIL;
HTable ←
DESCRIPTOR[AllocateHeapNode[hashlength*SIZE[DictEntry]],hashlength];
FOR i IN [0..hashlength) DO HTable[i] ← NIL; ENDLOOP;
LastProcessedEntry ← NIL;
Value ← 0;
END;
Lookup: PROCEDURE[key: ObjectName] RETURNS[CARDINAL] =
--Return value associated with key. If key not found then
--allocate a new entry with generated value, and return the value.
BEGIN
de: DictEntry;
h: CARDINAL;
[de,h] ← Where[key];
IF de=NIL THEN
{de ← AllocateHeapNode[SIZE[DictEntryRecord]];
de↑ ← [
next: NIL,
ovflow: HTable[h],
key: key,
value: (Value←Value+1)];
HTable[h] ← de;
IF DictTail=NIL THEN DictHead ← DictTail ← de
ELSE
{DictTail.next ← de;
DictTail ← de;
};
};
RETURN[de.value];
END;
GetNonProcessedEntry: PROCEDURE
RETURNS[key: ObjectName, value: CARDINAL] =
--Step LastProcessedEntry and return its key and value, if any
BEGIN
IF LastProcessedEntry=NIL THEN LastProcessedEntry ← DictHead
ELSE LastProcessedEntry ← LastProcessedEntry.next;
IF LastProcessedEntry=NIL THEN RETURN[NilObjectName,0]
ELSE RETURN[LastProcessedEntry.key, LastProcessedEntry.value];
END;
FreeDict: PROCEDURE =
BEGIN
de,nextde: DictEntry;
FOR de ← DictHead, nextde UNTIL de=NIL DO
nextde ← de.next;
FreeHeapNode[de];
ENDLOOP;
FreeHeapNode[BASE[HTable]];
END;
Where: PROCEDURE[key: ObjectName] RETURNS[de: DictEntry, h: CARDINAL] =
--h is result of hashing name
BEGIN
h ← Hash[key, LENGTH[HTable]];
FOR de ← HTable[h], de.ovflow UNTIL de=NIL OR key=de.key
DO --nothing-- ENDLOOP;
END;
Hash: PROCEDURE[key: ObjectName, range: CARDINAL] RETURNS[h: CARDINAL] =
--h is result of hashing name into [0..range)
BEGIN
i: CARDINAL ← LowHalf[key];
RETURN[i MOD range];
END;
--Utilities
WriteLongDecimal: PROCEDURE[n: LONG INTEGER] =
BEGIN
s: STRING ← [50];
AppendLongDecimal[s,n];
WriteString[s];
END;
WriteFloat: PROCEDURE[r: REAL] =
BEGIN
WriteReal[WriteChar,r];
END;
DotinName: PROCEDURE[name: STRING] RETURNS[BOOLEAN] =
BEGIN
FOR i:CARDINAL IN [0..name.length) DO
IF name[i]=’. THEN RETURN[TRUE];
ENDLOOP;
RETURN[FALSE];
END;
--CIF parameters
ScanningTopLevel: BOOLEAN;
CIFStream: StreamHandle ← NIL;
SaveLayer: CARDINAL;
CurrentLayer: CARDINAL;
Aborted: BOOLEAN;
LayerChar: ARRAY [0..MaxLENGTHLayerArray) OF CHARACTER ← ALL[’X];
LayerChar[0] ← ’I;--implant
LayerChar[1] ← ’D;--diffusion
LayerChar[2] ← ’P;--poly
LayerChar[3] ← ’C;--contact
LayerChar[4] ← ’M;--metal
LayerChar[5] ← ’B;--buried
LayerChar[6] ← ’G;--glass
LayerChar[7] ← ’X;--undef
--set up context
RegisterDevice[@CIFDeviceRecord];
END.