SisyphCmdsImpl.mesa
Copyright © 1985, 1986 by Xerox Corporation. All rights reserved.
Created by Pradeep Sindhu, December 9, 1985 10:01:52 pm PST
Pradeep Sindhu, September 26, 1986 2:06:15 pm PDT
Barth, January 13, 1986 3:30:05 pm PST
Bertrand Serlet March 4, 1987 7:16:45 pm PST
Jean-Marc Frailong June 20, 1986 8:25:11 pm PDT
Last Edited by: Jacobi July 15, 1986 2:40:48 pm PDT
DIRECTORY
Basics,
CD, CDBasics, CDCells, CDDirectory, CDImports, CDInstances, CDLayers, CDOps, CDPanelFonts, CDPopUpMenus, CDProperties, CDRects, CDSatellites, CDSequencer, CDTexts, CDViewer,
Core, CoreClasses, CoreOps, CoreGeometry, CoreProperties,
GList, IO, Rope, RopeList,
Sinix, SinixOps, Sisyph,
TerminalIO, ViewerClasses, ViewerOps;
SisyphCmdsImpl:
CEDAR
PROGRAM
IMPORTS Basics, CD, CDBasics, CDCells, CDDirectory, CDImports, CDInstances, CDLayers, CDOps, CDPanelFonts, CDPopUpMenus, CDProperties, CDRects, CDSatellites, CDSequencer, CDTexts, CDViewer, CoreClasses, CoreOps, CoreGeometry, CoreProperties, GList, IO, Rope, RopeList, Sinix, SinixOps, Sisyph, TerminalIO, ViewerOps
SHARES CDPopUpMenus, Sisyph = BEGIN
ROPE: TYPE = Rope.ROPE;
ROPES: TYPE = LIST OF ROPE;
Wire: TYPE = Core.Wire;
Wires: TYPE = Core.Wires;
CellType: TYPE = Core.CellType;
Icon: TYPE ~ REF IconRep;
IconRep:
TYPE ~
RECORD [
name: ROPE ← NIL, -- The short name of the icon, i.e. excluding the ".icon" part
boxWidth: CD.Number, -- Width of lines for icon box
pinRect: CD.Position, -- Size of pins (horizontal pin)
guard: CD.Number, -- Spacing between pin and pin name
grid: CD.Number, -- The grid on which pins should be aligned
font: CDTexts.CDFont, -- ChipNDale font to be used for all texts
pins: ARRAY CoreGeometry.Side OF ROPES -- The sorted pin names on all four sides
];
Icon Commands
UnMakeIcon:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
exprs: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelectedAndCell[selected, multiple] THEN RETURN;
FlushCache[selected.ob];
exprs ← NARROW [CDProperties.GetObjectProp[selected.ob, Sisyph.expressionsProp]];
CDProperties.PutObjectProp[selected.ob, Sisyph.expressionsProp, StripResultExprs[exprs]];
CDProperties.PutObjectProp[selected.ob, Sisyph.mode.extractProcProp, NIL];
CDProperties.PutObjectProp[selected.ob, $IconFor, NIL];
};
MakeCellIconWithCodeDef:
PROC [comm: CDSequencer.Command] = {
iconName: ROPE;
selected: CD.Instance;
multiple: BOOL;
cellIconExprPrefix: ROPE ← Rope.Cat[Sisyph.cellIconRope, " ← "];
cellIconExpr: ROPE;
exprs: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelectedAndCell[selected, multiple] THEN RETURN;
FlushCache[selected.ob];
iconName ← CDDirectory.Name[selected.ob];
IF NOT Rope.Match["*.icon", iconName] THEN TerminalIO.PutF["*** Convention for icons is to suffix them with '.icon'.\n"];
cellIconExpr ← TerminalIO.RequestRope[Rope.Cat["Type code: ", cellIconExprPrefix]];
cellIconExpr ← Rope.Cat[cellIconExprPrefix, cellIconExpr];
exprs ← NARROW [CDProperties.GetObjectProp[selected.ob, Sisyph.expressionsProp]];
exprs ← CONS [cellIconExpr, StripResultExprs[exprs]];
CDProperties.PutObjectProp[selected.ob, Sisyph.mode.extractProcProp, NIL];
CDProperties.PutObjectProp[selected.ob, Sisyph.expressionsProp, exprs];
CDProperties.PutObjectProp[selected.ob, $IconFor, NIL];
};
MakeCellIconWithGeometryDef:
PROC [comm: CDSequencer.Command] = {
iconName: ROPE;
selected: CD.Instance;
multiple: BOOL;
schematicName: ROPE;
cellIconExprPrefix: ROPE ← Rope.Cat[Sisyph.cellIconRope, " ← "];
cellIconExpr: ROPE;
exprs: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelectedAndCell[selected, multiple] THEN RETURN;
FlushCache[selected.ob];
iconName ← CDDirectory.Name[selected.ob];
IF NOT Rope.Match["*.icon", iconName] THEN TerminalIO.PutF["*** Convention for icons is to suffix them with '.icon'.\n"];
schematicName ← TerminalIO.RequestRope["Type schematic name: "];
IF CDDirectory.Fetch[comm.design, schematicName].object =
NIL
THEN {
TerminalIO.PutF["*** No such object in design.\n"];
RETURN;
};
IF NOT Rope.Match["*.sch", schematicName] THEN TerminalIO.PutF["*** Convention for schematics is to suffix them with '.sch'.\n"];
cellIconExpr ← Rope.Cat[cellIconExprPrefix, "ES[\"", schematicName, "\", cx]"];
exprs ← NARROW [CDProperties.GetObjectProp[selected.ob, Sisyph.expressionsProp]];
exprs ← CONS [cellIconExpr, StripResultExprs[exprs]];
CDProperties.PutObjectProp[selected.ob, Sisyph.mode.extractProcProp, NIL];
CDProperties.PutObjectProp[selected.ob, Sisyph.expressionsProp, exprs];
CDProperties.PutObjectProp[selected.ob, $IconFor, schematicName];
};
MakeWireIconWithCodeDef:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
wireIconExprPrefix: ROPE ← Rope.Cat[Sisyph.wireIconRope, " ← "];
wireIconExpr: ROPE;
exprs: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelectedAndCell[selected, multiple] THEN RETURN;
FlushCache[selected.ob];
wireIconExpr ← TerminalIO.RequestRope[Rope.Cat["Type code: ", wireIconExprPrefix]];
wireIconExpr ← Rope.Cat[wireIconExprPrefix, wireIconExpr];
exprs ← NARROW [CDProperties.GetObjectProp[selected.ob, Sisyph.expressionsProp]];
exprs ← CONS [wireIconExpr, StripResultExprs[exprs]];
CDProperties.PutObjectProp[selected.ob, Sisyph.mode.extractProcProp, NIL];
CDProperties.PutObjectProp[selected.ob, Sisyph.expressionsProp, exprs];
CDProperties.PutObjectProp[selected.ob, $IconFor, NIL];
};
CreateIconCommand:
PROC [comm: CDSequencer.Command] ~ {
User interface to create icon from schematic
schematicName, iconShortName: ROPE;
icon: CD.Object;
cell: Core.CellType;
selected: CD.Instance; multiple: BOOL;
grid: CD.Number ← -1;
cellRef: REF;
viewer: ViewerClasses.Viewer ← CDViewer.GetViewer[comm];
IF viewer#
NIL
THEN
WITH ViewerOps.GetViewer[viewer, $Grid]
SELECT
FROM
rgrid: REF CD.Number => grid ← rgrid^;
ENDCASE => NULL;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelectedAndCell[selected, multiple] THEN RETURN;
schematicName ← CDDirectory.Name[selected.ob];
IF schematicName=
NIL
THEN {
TerminalIO.PutF["*** Selected schematic has no name.\n"];
RETURN;
};
IF NOT Rope.Match["*.sch", schematicName] THEN TerminalIO.PutF["*** Convention for schematics is to suffix them with '.sch'.\n"];
cellRef ← SinixOps.ExtractCDInstance[selected, comm.design, Sisyph.mode].result;
IF ISTYPE [cellRef, Core.CellType] THEN cell ← NARROW [cellRef]
ELSE {
TerminalIO.PutF["*** Selected cell does not extract to Core CellType\n"];
RETURN;
};
iconShortName ← TerminalIO.RequestRope["Type icon short name: "];
IF Rope.IsEmpty[iconShortName]
THEN iconShortName ←
IF Rope.Match["*.sch", schematicName]
THEN Rope.Substr[schematicName, 0, Rope.Length[schematicName]-4]
ELSE schematicName;
IF Rope.IsEmpty[schematicName]
THEN {
TerminalIO.PutF["No name provided, no default from schematic.\n"];
RETURN;
};
IF CDDirectory.Fetch[comm.design, Rope.Cat[iconShortName, ".icon"]].found
THEN {
TerminalIO.PutF["*** The icon %g.icon already exists!\n", IO.rope[iconShortName]];
RETURN;
};
icon ← IconFromSchematic[cell, schematicName, iconShortName, comm.design, grid];
IF icon=NIL THEN RETURN;
CDCells.SetSimplificationTreshhold[cell: icon, val: 30, inPixels: TRUE];
CDOps.IncludeObjectI[comm.design, icon, comm.pos];
};
MakeSequenceIcon:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance; multiple: BOOL;
exprs: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelectedAndCell[selected, multiple] THEN RETURN;
IF
NOT CDCells.IsCell[selected.ob]
THEN {
TerminalIO.PutF["*** Selected object is not a cell.\n"];
RETURN;
};
IF
NOT ContainsSequenceResult[CDSatellites.GetSatelliteRopes[selected.ob]]
THEN {
TerminalIO.PutF["*** Warning: there is no satellite of the form 'Keyword: Expression', where 'Keyword' is one of the following:"];
FOR i:
NAT
IN [0 .. nbSequenceKeyWords)
DO TerminalIO.PutF[" %g", IO.rope[sequenceKeyWords[i]]] ENDLOOP;
TerminalIO.PutF["\n"];
} ELSE
TerminalIO.PutF["Sequencing of %g done.\n", IO.rope[CDDirectory.Name[selected.ob]]];
FlushCache[selected.ob];
exprs ← NARROW [CDProperties.GetObjectProp[selected.ob, Sisyph.expressionsProp]];
CDProperties.PutObjectProp[selected.ob, Sisyph.expressionsProp, StripResultExprs[exprs]];
CDProperties.PutObjectProp[selected.ob, Sisyph.mode.extractProcProp, $SisyphExtractSequence];
CDProperties.PutObjectProp[selected.ob, $IconFor, NIL];
};
Other Commands
MakeInvisibleToExtractor:
PROC [comm: CDSequencer.Command] = {
nInstances: INT ← 0;
FOR w:
LIST
OF
CD.Instance ← CDOps.InstList[comm.design], w.rest
WHILE w#
NIL
DO
inst: CD.Instance ← w.first;
IF inst.selected
THEN {
CDProperties.PutInstanceProp[inst, Sisyph.mode.extractProcProp, $ExtractNull];
nInstances ← nInstances+1;
FlushCache[inst.ob]
}
ENDLOOP;
TerminalIO.PutF["Made %g instances invisible to Sisyph\n", IO.int[nInstances]]
};
MakeVisibleToExtractor:
PROC [comm: CDSequencer.Command] = {
nInstances: INT ← 0;
FOR w:
LIST
OF
CD.Instance ← CDOps.InstList[comm.design], w.rest
WHILE w#
NIL
DO
inst: CD.Instance ← w.first;
IF inst.selected
THEN {
CDProperties.PutInstanceProp[inst, Sisyph.mode.extractProcProp, NIL];
nInstances ← nInstances+1;
FlushCache[inst.ob]
}
ENDLOOP;
TerminalIO.PutF["Made %g instances visible to Sisyph\n", IO.int[nInstances]]
};
FlushCache:
PROC [obj:
CD.Object] = {
FlushEachInstance: CDCells.InstEnumerator = {FlushCache[inst.ob]};
Sinix.FlushCache[obj];
IF CDCells.IsCell[obj] THEN [] ← CDCells.EnumerateInstances[obj, FlushEachInstance];
};
FlushSisyphCaches:
PROC [comm: CDSequencer.Command] = {
FOR w:
LIST
OF
CD.Instance ← CDOps.InstList[comm.design], w.rest
WHILE w#
NIL
DO
IF w.first.selected THEN FlushCache[w.first.ob];
ENDLOOP;
};
ShowInstExpressions:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelected[selected, multiple] THEN RETURN;
TerminalIO.PutF["Instance Expressions:\n"];
PutRopes[NARROW [CDProperties.GetInstanceProp[selected, Sisyph.expressionsProp]]];
};
AddInstExpression:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
exprs: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelected[selected, multiple] THEN RETURN;
TerminalIO.PutF["Add Instance Expression\n"];
exprs ← NARROW [CDProperties.GetInstanceProp[selected, Sisyph.expressionsProp]];
exprs ← CONS [TerminalIO.RequestRope[Rope.Cat[" New Expression: "]], exprs];
CDProperties.PutInstanceProp[selected, Sisyph.expressionsProp, exprs]
};
EditInstExpressions:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
exprs: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelected[selected, multiple] THEN RETURN;
TerminalIO.PutF["Edit Instance Expressions\n"];
exprs ← NARROW[CDProperties.GetInstanceProp[selected, Sisyph.expressionsProp]];
exprs ← EditExpressions[exprs, "Expression"];
CDProperties.PutInstanceProp[selected, Sisyph.expressionsProp, exprs]
};
ShowObjExpressions:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelected[selected, multiple] THEN RETURN;
TerminalIO.PutF["Object Expressions:\n"];
PutRopes[NARROW [CDProperties.GetObjectProp[selected.ob, Sisyph.expressionsProp]]];
IF NOT CDImports.IsImport[selected.ob] THEN RETURN;
IF
NARROW [selected.ob.specific, CDImports.ImportSpecific].boundInstance=
NIL
THEN {TerminalIO.PutF["Object is an unbound import.\n"]; RETURN};
TerminalIO.PutF["Object Expressions of the bound import:\n"];
PutRopes[NARROW [CDProperties.GetObjectProp[NARROW [selected.ob.specific, CDImports.ImportSpecific].boundInstance.ob, Sisyph.expressionsProp]]];
};
AddObjExpression:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
exprs: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelected[selected, multiple] THEN RETURN;
TerminalIO.PutF["Add Object Expression\n"];
exprs ← NARROW [CDProperties.GetObjectProp[selected.ob, Sisyph.expressionsProp]];
exprs ← CONS [TerminalIO.RequestRope[Rope.Cat[" New Expression: "]], exprs];
CDProperties.PutObjectProp[selected.ob, Sisyph.expressionsProp, exprs]
};
EditObjExpressions:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
exprs: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelected[selected, multiple] THEN RETURN;
TerminalIO.PutF["Edit Object Expressions\n"];
exprs ← NARROW[CDProperties.GetObjectProp[selected.ob, Sisyph.expressionsProp]];
exprs ← EditExpressions[exprs, "Expression"];
CDProperties.PutObjectProp[selected.ob, Sisyph.expressionsProp, exprs]
};
ShowParmNames:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
parms: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelected[selected, multiple] THEN RETURN;
TerminalIO.PutF["Parm Names:"];
parms ← NARROW[CDProperties.GetObjectProp[selected.ob, Sisyph.parmNamesProp]];
WHILE parms#
NIL
DO
TerminalIO.PutF[Rope.Cat["\n ", parms.first]];
parms ← parms.rest;
ENDLOOP;
TerminalIO.PutF["\n"];
};
AddParmName:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
parms: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelected[selected, multiple] THEN RETURN;
TerminalIO.PutF["Add Parameter Name\n"];
parms ← NARROW [CDProperties.GetObjectProp[selected.ob, Sisyph.parmNamesProp]];
parms ← CONS [TerminalIO.RequestRope[Rope.Cat[" New Parameter: "]], parms];
CDProperties.PutObjectProp[selected.ob, Sisyph.parmNamesProp, parms]
};
EditParmNames:
PROC [comm: CDSequencer.Command] = {
selected: CD.Instance;
multiple: BOOL;
parms: ROPES;
[selected, multiple] ← CDOps.SelectedInstance[comm.design];
IF ~IsSingleSelected[selected, multiple] THEN RETURN;
TerminalIO.PutF["Edit Object Expressions\n"];
parms ← NARROW[CDProperties.GetObjectProp[selected.ob, Sisyph.parmNamesProp]];
parms ← EditExpressions[parms, "Parameter"];
CDProperties.PutObjectProp[selected.ob, Sisyph.parmNamesProp, parms]
};
Create icon from schematic
AlignUpToGrid:
PROC [x, grid:
CD.Number]
RETURNS [
CD.Number] ~ {
RETURN [IF x>=0 THEN ((x+grid-1)/grid)*grid ELSE (x/grid)*grid]
};
AlignDownToGrid:
PROC [x, grid:
CD.Number]
RETURNS [
CD.Number] ~ {
RETURN [IF x>=0 THEN (x/grid)*grid ELSE ((x-grid+1)/grid)*grid]
};
AlignPositionUpToGrid:
PROC [p:
CD.Position, grid:
CD.Number]
RETURNS [
CD.Position] ~ {
RETURN [[AlignUpToGrid[p.x, grid], AlignUpToGrid[p.y, grid]]]
};
AlignRectUpToGrid:
PROC [r:
CD.Rect, grid:
CD.Number]
RETURNS [
CD.Rect] ~ {
RETURN [[x1: AlignDownToGrid[r.x1, grid], x2: AlignUpToGrid[r.x2, grid], y1: AlignDownToGrid[r.y1, grid], y2: AlignUpToGrid[r.y2, grid]]]
};
MoveInstances:
PROC [list:
CD.InstanceList, by:
CD.Position] ~ {
Move all instances in the list by the required amount
WHILE list#
NIL
DO
list.first.trans.off ← CDBasics.AddPoints[list.first.trans.off, by];
list ← list.rest;
ENDLOOP;
};
BuildIconSides:
PROC [icon: Icon]
RETURNS [sideSize:
ARRAY CoreGeometry.Side
OF
CD.Position, sideList:
ARRAY CoreGeometry.Side
OF
CD.InstanceList] ~ {
Build all the pins and pin names for the icon.
BuildSideGeometry:
PROC [on: CoreGeometry.Side, rotate:
CD.Orientation, useTextEnd:
BOOL, textOffset:
CD.Position, pinStartPos:
CD.Position] ~ {
Create geometry (pins and names) along an edge of the icon. All pins are aligned on grid coordinates. Text is centered on associated pin, at left or right according to request. Text is separated by guard CD units from the pin. The returned iLists alternates pins and names.
iList: CD.InstanceList ← NIL;
pinPos, satPos: CD.Position; -- real ones
pinPos ← pinStartPos;
FOR names:
ROPES ← icon.pins[on], names.rest
WHILE names#
NIL
DO
satObject: CD.Object = CDTexts.Create[text: names.first, font: icon.font];
size: CD.Position = CDBasics.SizeOfRect[satObject.bbox];
pinObject: CD.Object = CDRects.CreateRect[size: icon.pinRect, l: CD.commentLayer];
satInst, pinInst: CD.Instance;
pinInst ← CDInstances.NewInst[pinObject, [pinPos, rotate]];
satPos ← CDBasics.AddPoints[pinPos, textOffset];
satPos ← CDBasics.SubPoints[
satPos,
IF CDBasics.IncludesOddRot90[rotate]
THEN [size.y/2, IF useTextEnd THEN 0 ELSE -size.x]
ELSE [IF useTextEnd THEN size.x ELSE 0, size.y/2]
];
satInst ← CDInstances.NewInst[satObject, [satPos, rotate]];
Progress pin position
pinPos ← AlignPositionUpToGrid[CDBasics.AddPoints[pinPos, CDBasics.OrientedSize[[0, size.y], rotate]], icon.grid];
Set satellite property and add to result list
CDSatellites.Associate[master: pinInst, text: satInst];
iList ← CONS[pinInst, iList];
iList ← CONS[satInst, iList];
ENDLOOP;
IF iList#
NIL
THEN {
-- Compute size (not default) and set logical base at [0,0]
r: CD.Rect = AlignRectUpToGrid[CDInstances.BoundingRectO[iList], icon.grid];
pos: CD.Position = CDBasics.BaseOfRect[r];
sideSize[on] ← CDBasics.SizeOfRect[r];
MoveInstances[iList, CDBasics.NegOffset[pos]];
}
ELSE sideSize[on] ← [0, 0];
sideList[on] ← iList;
};
BuildSideGeometry[on: top,
rotate: rotate270,
useTextEnd: TRUE,
textOffset: [0, -icon.pinRect.x-icon.guard],
pinStartPos: [0, -icon.pinRect.x]];
BuildSideGeometry[on: bottom,
rotate: rotate270,
useTextEnd: FALSE,
textOffset: [0, icon.pinRect.x-icon.guard],
pinStartPos: [0, 0]];
BuildSideGeometry[on: right,
rotate: original,
useTextEnd: TRUE,
textOffset: [-icon.guard, 0],
pinStartPos: [-icon.pinRect.x, 0]];
BuildSideGeometry[on: left,
rotate: original,
useTextEnd: FALSE,
textOffset: [icon.pinRect.x+icon.guard, 0],
pinStartPos: [0, 0]];
};
BuildIconFrame:
PROC [icon: Icon, box:
CD.Rect]
RETURNS [list:
CD.InstanceList ←
NIL] ~ {
Build the geometry for the icon box
BuildBoxSide:
PROC [size, at:
CD.Position, orient:
CD.Orientation] ~ {
list ← CONS [CDInstances.NewInst[CDRects.CreateRect[size: size, l: CD.commentLayer], [at, orient]], list];
CDProperties.PutInstanceProp[list.first, Sisyph.mode.extractProcProp, $ExtractNull];
};
w: CD.Number = icon.boxWidth;
BuildBoxSide[size: [w, w+box.y2-box.y1], at: [box.x1, box.y1], orient: original]; -- left
BuildBoxSide[size: [w, w+box.y2-box.y1], at: [box.x2, box.y1], orient: original]; -- right
BuildBoxSide[size: [w, w+box.x2-box.x1], at: [box.x2+w, box.y1], orient: rotate90]; -- bottom
BuildBoxSide[size: [w, w+box.x2-box.x1], at: [box.x2+w, box.y2], orient: rotate90]; -- top
};
CommentText:
PROC [text:
ROPE, font: CDTexts.CDFont]
RETURNS [ob:
CD.Object] ~ {
IsItalic:
PROC [fontName:
ROPE]
RETURNS [
BOOL] = {
RETURN [Rope.Find[fontName, "i", Rope.Length[fontName]-1, FALSE]#-1]
};
italicFont: CDTexts.CDFont ←
IF ~IsItalic[font.supposedName]
THEN CDTexts.MakeFont[
name: Rope.Cat[font.supposedName, "I"],
scale: font.scaleI]
ELSE font;
ob ← CDTexts.Create[text: text, font: italicFont];
};
BuildIconObject:
PROC [icon: Icon, design:
CD.Design]
RETURNS [obj:
CD.Object] ~ {
Build the CD object for the icon in the design. The object is included in the design. Raises an ERROR if the object already existed (should be checked by caller).
sideSize: ARRAY CoreGeometry.Side OF CD.Position;
sideList: ARRAY CoreGeometry.Side OF CD.InstanceList;
internalInstances: CD.InstanceList ← NIL;
iconFullName: ROPE = Rope.Cat[icon.name, ".icon"];
iconTextObject: CD.Object = CommentText[text: icon.name, font: icon.font];
size: CD.Position = CDBasics.SizeOfRect[iconTextObject.bbox];
iconText: CD.Instance;
inside, outside: CD.Rect;
width, height: CD.Number;
IF icon.grid<=0 THEN icon.grid ← design.technology.lambda;
[sideSize, sideList] ← BuildIconSides[icon];
Compute a reasonable size for the inside and outside of the icon.
width ← sideSize[left].x+sideSize[right].x+
MAX[
sideSize[bottom].x,
sideSize[top].x,
AlignUpToGrid[2*size.x, icon.grid]];
height ←
MAX[
sideSize[left].y,
sideSize[right].y,
sideSize[bottom].y+sideSize[top].y+AlignUpToGrid[4*size.y, icon.grid]];
inside.x1 ← sideSize[left].x;
inside.y1 ← sideSize[bottom].y;
inside.x2 ← width+icon.pinRect.y-sideSize[right].x;
inside.y2 ← height+icon.pinRect.y-sideSize[top].y;
outside.x1 ← IF sideList[left]=NIL THEN 0 ELSE icon.pinRect.x;
outside.y1 ← IF sideList[bottom]=NIL THEN 0 ELSE icon.pinRect.x;
outside.x2 ← IF sideList[right]=NIL THEN width ELSE width-icon.pinRect.x;
outside.y2 ← IF sideList[top]=NIL THEN height ELSE height-icon.pinRect.x;
Move the sides at their final locations and setup the icon name
MoveInstances[
sideList[bottom],
[AlignDownToGrid[(width+sideSize[left].x-sideSize[right].x-sideSize[bottom].x)/2, icon.grid], 0]];
MoveInstances[
sideList[top],
[AlignDownToGrid[(width+sideSize[left].x-sideSize[right].x-sideSize[top].x)/2, icon.grid], inside.y2]];
MoveInstances[
sideList[left],
[0, AlignDownToGrid[(height-sideSize[left].y)/2, icon.grid]]];
MoveInstances[
sideList[right],
[inside.x2, AlignDownToGrid[(height-sideSize[right].y)/2, icon.grid]]];
iconText ← CDInstances.NewInst[
iconTextObject,
[CDBasics.SubPoints[CDBasics.Center[inside], CDBasics.Center[CDBasics.RectAt[[0, 0], size]]]]];
Build the icon object
internalInstances ← BuildIconFrame[icon, outside];
internalInstances ← CONS [iconText, internalInstances];
FOR on: CoreGeometry.Side
IN CoreGeometry.Side
DO
FOR list:
CD.InstanceList ← sideList[on], list.rest
WHILE list#
NIL
DO
internalInstances ← CONS [list.first, internalInstances];
ENDLOOP;
ENDLOOP;
obj ← CDCells.CreateCell[il: internalInstances]; -- of course, one would really like a sequence!
IF CDDirectory.Fetch[design, iconFullName].found
THEN {
TerminalIO.PutF["*** The icon %g already exists!\n", IO.rope[iconFullName]];
ERROR;
};
IF
NOT CDDirectory.Include[design, obj, iconFullName]
THEN {
TerminalIO.PutF["*** Direction insertion of icon %g failed.\n", IO.rope[iconFullName]];
ERROR;
};
};
IconFromSchematic:
PROC [cell: Core.CellType, schematicName, iconName:
ROPE, design:
CD.Design, grid:
CD.Number ← -1]
RETURNS [obj:
CD.Object] ~ {
Build an icon straight from the schematic. Return NIL object if error. The schematicName had better be the right one.
InternalPin: TYPE ~ REF InternalPinRep;
InternalPinRep:
TYPE ~
RECORD [
name: ROPE,
rect: CD.Rect -- The InstRectO of the pin, used to sort along the side
];
Side:
TYPE ~
RECORD [
border: CD.Rect, -- Representation of schematic side
pins: LIST OF InternalPin ← NIL
];
SearchPins: CoreOps.EachWireProc ~ {
pinGeom: CoreGeometry.Instances ← CoreGeometry.GetPins[Sisyph.mode.decoration, wire];
pinName: ROPE = CoreOps.GetShortWireName[wire];
IF pinGeom=NIL THEN RETURN;
IF pinName=
NIL
THEN {
TerminalIO.PutF["*** The wire %g has pins, but no short name\n", IO.rope[CoreOps.GetFullWireName[root: cell.public, wire: wire]]];
RETURN[quit: TRUE];
};
WHILE pinGeom#
NIL
DO
-- Paranoid! the first pin should touch a side of the schematic
pinRect: CD.Rect = CoreGeometry.BBox[pinGeom.first];
FOR on: CoreGeometry.Side
IN CoreGeometry.Side
DO
IF CDBasics.Intersect[pinRect, sides[on].border]
THEN {
sides[on].pins ← CONS [NEW [InternalPinRep ← [name: pinName, rect: pinRect]], sides[on].pins];
RETURN [subWires: FALSE];
};
ENDLOOP;
pinGeom ← pinGeom.rest;
ENDLOOP;
TerminalIO.PutF["*** The wire %g has pins, but none of them touches the icon of the schematic.\n", IO.rope[CoreOps.GetFullWireName[root: cell.public, wire: wire]]];
RETURN [quit: TRUE];
};
XSort: GList.CompareProc ~ {
-- Reverse order comparison
RETURN [Basics.CompareINT[NARROW[ref2, InternalPin].rect.x1, NARROW[ref1, InternalPin].rect.x1]];
};
YSort: GList.CompareProc ~ {
-- Reverse order comparison
RETURN [Basics.CompareINT[NARROW[ref2, InternalPin].rect.y1, NARROW[ref1, InternalPin].rect.y1]];
};
lineWidth: CD.Number = CDLayers.LayerWidth[design, CD.commentLayer];
icon: Icon ← NEW [IconRep ← [name: iconName, boxWidth: lineWidth, pinRect: [x: 4*lineWidth, y: lineWidth], guard: 2*lineWidth, grid: grid, font: CDPanelFonts.CurrentFont[design]]];
sides: ARRAY CoreGeometry.Side OF Side;
cellIR: CD.Rect = CD.InterestRect[CoreGeometry.GetObject[Sisyph.mode.decoration, cell]];
exprs: ROPES ← NIL;
First, build rectangles representing the sides of the schematic
sides[top].border ← [x1: cellIR.x1, x2: cellIR.x2, y1: cellIR.y2, y2: cellIR.y2];
sides[bottom].border ← [x1: cellIR.x1, x2: cellIR.x2, y1: cellIR.y1, y2: cellIR.y1];
sides[right].border ← [x1: cellIR.x2, x2: cellIR.x2, y1: cellIR.y1, y2: cellIR.y2];
sides[left].border ← [x1: cellIR.x1, x2: cellIR.x1, y1: cellIR.y1, y2: cellIR.y2];
Scan the public to build the sides structure
IF CoreOps.VisitWireSeq[cell.public, SearchPins]
THEN
RETURN [NIL];
Sort the pins on each side and setup the names in the icon structure
FOR on: CoreGeometry.Side
IN CoreGeometry.Side
DO
icon.pins[on] ← NIL;
FOR list:
LIST
OF InternalPin ←
NARROW [GList.UniqueSort[list: sides[on].pins, compareProc:
IF on=top
OR on=bottom
THEN XSort
ELSE YSort]], list.rest
WHILE list#
NIL
DO
icon.pins[on] ← CONS [list.first.name, icon.pins[on]];
ENDLOOP;
ENDLOOP;
Finally create the object from the icon structure
obj ← BuildIconObject[icon, design];
exprs ← CONS [Rope.Cat[Sisyph.cellIconRope, " ← ES[\"", schematicName, "\", cx]"], exprs];
CDProperties.PutObjectProp[obj, Sisyph.mode.extractProcProp, NIL];
CDProperties.PutObjectProp[obj, Sisyph.expressionsProp, exprs];
CDProperties.PutObjectProp[obj, $IconFor, schematicName];
};
Sequence Icons
nbSequenceKeyWords: NAT = 5;
sequenceKeyWords: ARRAY [0 .. nbSequenceKeyWords) OF ROPE = ["Seq", "SeqX", "SeqY", "RSeqX", "RSeqY"];
sequenceLayoutAtoms:
ARRAY [0 .. nbSequenceKeyWords)
OF
ATOM = [
NIL, $ArrayX, $ArrayY, $ReverseArrayY, $ReverseArrayY];
-- At this point, there is a strange dependency between Sisyph and PWCore, but it is not worth the effort making a separate module
FindMatchingKeyword:
PROC [expr:
ROPE]
RETURNS [keyword:
NAT ← nbSequenceKeyWords, sequencing:
ROPE ←
NIL] = {
FOR i:
NAT
IN [0 .. nbSequenceKeyWords)
DO
IF Rope.Match[Rope.Cat[sequenceKeyWords[i], ":*"], expr] THEN RETURN [i, Rope.Substr[expr, Rope.Find[expr, ":"]+1]];
ENDLOOP;
};
IsSequenceResult:
PROC [expr:
ROPE]
RETURNS [
BOOL] = {
RETURN [FindMatchingKeyword[expr].keyword#nbSequenceKeyWords]
};
ContainsSequenceResult:
PROC [exprs:
ROPES]
RETURNS [
BOOL ←
FALSE] = {
WHILE exprs#
NIL
DO
IF IsSequenceResult[exprs.first] THEN RETURN [TRUE]; exprs ← exprs.rest ENDLOOP;
};
FindPorts:
PROC [baseCell: CellType, wires: Wires]
RETURNS [set: CoreClasses.SequenceSet] = {
nats: LIST OF NAT ← NIL;
size: NAT ← 0;
WHILE wires#
NIL
DO
FOR w:
NAT
IN [0 .. baseCell.public.size)
DO
sequenceName: ROPE ← CoreOps.GetShortWireName[wires.first];
IF wires.first=baseCell.public[w]
OR (sequenceName#
NIL
AND Rope.Equal[sequenceName, CoreOps.GetShortWireName[baseCell.public[w]]])
THEN {
nats ← CONS [w, nats]; size ← size + 1;
EXIT;
};
REPEAT FINISHED => ERROR;
ENDLOOP;
wires ← wires.rest;
ENDLOOP;
set ← NEW [CoreClasses.SequenceSetRec[size]];
FOR i: INT IN [0 .. size) DO set[i] ← nats.first; nats ← nats.rest ENDLOOP;
};
ExtractSequence: Sinix.ExtractProc = {
AddProp:
PROC [prop:
ATOM, val:
REF
ANY] = {
CoreProperties.PutCellTypeProp[cellType, prop, val];
};
cx: Sisyph.Context;
resultRope: ROPE;
cellType: CellType;
name: ROPE = CDDirectory.Name[obj];
keyword: NAT; sequencing: ROPE; count: NAT;
[cx, resultRope] ← Sisyph.EvaluateParameters[userData, obj, properties, IsSequenceResult];
IF resultRope=
NIL
THEN {
TerminalIO.PutF["*** SisyphExtractSequence: Sequence does not contain any of sequencing information (e.g. an object satellite 'SeqX: 32').\n"];
ERROR;
};
[keyword, sequencing] ← FindMatchingKeyword[resultRope];
Sisyph.Eval[cx, sequencing];
count ← NAT [Sisyph.FetchInt[cx, "&"].value];
TerminalIO.PutF["Extracting [Sisyph] cell %g (%g: %g)\n", IO.rope[CDDirectory.Name[obj]], IO.rope[sequenceKeyWords[keyword]], IO.int[count]];
cellType ← ExtractSequenceIcon[obj, cx, keyword, count];
props ← Sisyph.GetCoreInstProps[cx];
CoreProperties.Enumerate[Sisyph.GetCoreProps[cx], AddProp];
IF CoreOps.GetCellTypeName[cellType]=NIL AND name#NIL THEN [] ← CoreOps.SetCellTypeName[cellType, Rope.Substr[name, 0, Rope.Index[name, 0, "."]]];
result ← cellType;
};
ExtractSequenceIcon:
PROC [obj:
CD.Object, cx: Sisyph.Context, keyword, count:
NAT]
RETURNS [sequence: CellType] = {
iconCT: CellType = NARROW [Sinix.ExtractCell[obj, Sisyph.mode, NIL, cx].result];
iconRCT: CoreClasses.RecordCellType = NARROW [iconCT.data];
subCT: CellType;
sequenceWires, flatSequenceWires: Wires ← NIL;
There should be only one subcell
IF iconRCT.size#1
THEN {
TerminalIO.PutF["*** SisyphExtractSequence: Sequence should contain one and only one subcell.\n"];
ERROR;
};
subCT ← iconRCT[0].type;
We deal with Global Variables
Sisyph.ProcessGlobalNames[iconCT, cx];
we check that there is no internal only
FOR i:
NAT
IN [0 .. iconRCT.internal.size)
DO
wire: Wire = iconRCT.internal[i];
name: ROPE ← CoreOps.GetShortWireName[wire];
IF name=NIL THEN name ← "some wire";
IF
NOT CoreOps.RecursiveMember[iconRCT[0].actual, wire]
THEN {
TerminalIO.PutF["*** SisyphExtractSequence: %g is not connected to subcell.\n", IO.rope[name]];
ERROR;
};
IF
NOT CoreOps.RecursiveMember[iconCT.public, wire]
THEN {
TerminalIO.PutF["*** SisyphExtractSequence: %g is not public.\n", IO.rope[name]];
ERROR;
};
ENDLOOP;
We compute which wires are going to be sequenced
FOR i:
NAT
IN [0 .. subCT.public.size)
DO
IF CoreProperties.GetWireProp[iconRCT[0].actual[i], $Sequence]#
NIL
THEN sequenceWires ← CONS [subCT.public[i], sequenceWires];
IF CoreProperties.GetWireProp[iconRCT[0].actual[i], $FlatSequence]#
NIL
THEN flatSequenceWires ← CONS [subCT.public[i], flatSequenceWires];
ENDLOOP;
We create the sequence
sequence ← CoreClasses.CreateSequence[NEW [CoreClasses.SequenceCellTypeRec ← [base: subCT, count: count, sequence: FindPorts[subCT, sequenceWires], flatSequence: FindPorts[subCT, flatSequenceWires]]]];
We copy the pins decorations
FOR i:
NAT
IN [0 .. sequence.public.size)
DO
iconWire: Wire = CoreClasses.CorrespondingActual[iconRCT[0], subCT.public[i]];
IF iconWire=NIL THEN ERROR;
CoreGeometry.PutIndirectLazyPins[Sisyph.mode.decoration, sequence.public[i], iconWire];
CoreGeometry.PutGeometry[Sisyph.mode.decoration, sequence.public[i], CoreGeometry.GetGeometry[Sisyph.mode.decoration, iconWire]];
ENDLOOP;
The object decoration!
CoreGeometry.PutObject[Sisyph.mode.decoration, sequence, obj];
We decorate with the appriopriate layout atom
CoreProperties.PutCellTypeProp[sequence, $Layout, sequenceLayoutAtoms[keyword]];
};
Initialization
FilterSch: SinixOps.FilterProc = {RETURN [Rope.Match["*.sch", name] OR Rope.Match["*.icon", name]]};
Main Menu
Sinix.RegisterExtractProc[$SisyphExtractSequence, ExtractSequence];
CDSequencer.ImplementCommand[key: $FlushSisyphCaches, proc: FlushSisyphCaches];
SinixOps.RegisterModeCommands[mode: Sisyph.mode];
SinixOps.RegisterBackgroundExtractionCommand[CD.FetchTechnology[$cmosB], Sisyph.mode, "Sch background extraction", $SchBackgroundExtract, FilterSch]; -- only works for CMosB. Not very clean
Make Icon Menu
[] ← CDPopUpMenus.MakeMenu[$SisyphIconMenu, "Sisyph: Make Icon", NIL];
CDSequencer.ImplementCommand[key: $CreateIconCommand, proc: CreateIconCommand];
CDSequencer.ImplementCommand[key: $MakeCellIconWithGeometryDef, proc: MakeCellIconWithGeometryDef];
CDSequencer.ImplementCommand[key: $MakeCellIconWithCodeDef, proc: MakeCellIconWithCodeDef];
CDSequencer.ImplementCommand[key: $MakeSequenceIcon, proc: MakeSequenceIcon];
CDSequencer.ImplementCommand[key: $MakeWireIconWithCodeDef, proc: MakeWireIconWithCodeDef];
CDSequencer.ImplementCommand[key: $UnMakeIcon, proc: UnMakeIcon];
Change Visibility Menu
[] ← CDPopUpMenus.MakeMenu[$SisyphVisibilityMenu, "Sisyph: Change Visibility", NIL];
CDSequencer.ImplementCommand[key: $MakeInvisibleToExtractor, proc: MakeInvisibleToExtractor];
CDSequencer.ImplementCommand[key: $MakeVisibleToExtractor, proc: MakeVisibleToExtractor];
Make Parameter Menu (normally hidden)
[] ← CDPopUpMenus.MakeMenu[$SisyphParameterMenu, "Sisyph: Parameter Specification", NIL];
CDSequencer.ImplementCommand[key: $SisyphShowParmNames, proc: ShowParmNames, queue: doQueue];
CDSequencer.ImplementCommand[key: $SisyphAddParmName, proc: AddParmName];
CDSequencer.ImplementCommand[key: $SisyphEditParmNames, proc: EditParmNames];
Expressions Menu
CDSequencer.ImplementCommand[key: $SisyphShowInstExpressions, proc: ShowInstExpressions, queue: doQueue];
CDSequencer.ImplementCommand[key: $SisyphAddInstExpression, proc: AddInstExpression];
CDSequencer.ImplementCommand[key: $SisyphEditInstExpressions, proc: EditInstExpressions];
CDSequencer.ImplementCommand[key: $SisyphShowObjExpressions, proc: ShowObjExpressions, queue: doQueue];
CDSequencer.ImplementCommand[key: $SisyphAddObjExpression, proc: AddObjExpression];
CDSequencer.ImplementCommand[key: $SisyphEditObjExpressions, proc: EditObjExpressions];
END.