PipalInstanceTableImpl.mesa
Copyright Ó 1987, 1988 by Xerox Corporation. All rights reversed.
From Intervals, created by Bertrand Serlet, November 16, 1985 7:22:05 pm PST
Bertrand Serlet, March 15, 1988 0:36:02 am PST
Jean-Marc Frailong January 16, 1988 5:30:56 pm PST
DIRECTORY
Basics USING [LongNumber, BITSHIFT, BITOR, BITAND],
Pipal, PipalInt, PipalInstanceTable;
PipalInstanceTableImpl: CEDAR MONITOR
IMPORTS Basics, PipalInt
EXPORTS PipalInstanceTable =
BEGIN OPEN PipalInstanceTable;
freeInstanceValues: LIST OF InstanceValue ← NIL;
Hand allocation for efficiency reasons
equivalent to: new ← CONS [iw, list];
Cons: PRIVATE PROC [iw: InstanceValue, list: LIST OF InstanceValue] RETURNS [new: LIST OF InstanceValue] = {
IF freeInstanceValues=NIL THEN freeInstanceValues ← LIST [[[], NIL]]; -- too bad!
IF freeInstanceValues.first#[[], NIL] THEN ERROR; -- something real bad going on!
new ← freeInstanceValues; freeInstanceValues ← freeInstanceValues.rest; new.first ← iw; new.rest ← list;
};
equivalent to: rest ← list.rest;
only disposes of the first element!
Dispose: PRIVATE PROC [list: LIST OF InstanceValue] RETURNS [rest: LIST OF InstanceValue] = {
rest ← list.rest; list.first ← [[], NIL]; list.rest ← freeInstanceValues; freeInstanceValues ← list;
IF freeInstanceValues.first#[[], NIL] THEN ERROR; -- something real bad going on!
};
Range: TYPE = RECORD [x1, y1, x2, y2: INT];
For efficiency of intersection.
Halve: PROC [n: INT] RETURNS [INT] ~ INLINE {
Because the compiler is brain-damaged and calls general division for INT/2
ln: Basics.LongNumber;
ln.li ← n;
ln.lo ← Basics.BITOR[Basics.BITSHIFT[ln.lo, -1], Basics.BITSHIFT[ln.hi, 15]];
ln.hi ← Basics.BITOR[Basics.BITSHIFT[ln.hi, -1], Basics.BITAND[ln.hi, 08000H]];
RETURN [ln.li];
};
Intersect: PROC [r1, r2: Range] RETURNS [BOOL] = INLINE {
Returns r1 and r2 have some common points or border; [assumes r1, r2 normalized].
[TRUE if rects touch on a single point]
RETURN [r1.x1<=r2.x2 AND r2.x1<=r1.x2 AND r1.y1<=r2.y2 AND r2.y1<=r1.y2]
};
Inside: PROC [a, b: Range] RETURNS [BOOL] = INLINE {
Returns a inside of b (b inclusive border)
RETURN[a.x1>=b.x1 AND a.x2<=b.x2 AND a.y1>=b.y1 AND a.y2<=b.y2]
};
Intersection: PROC [r1, r2: Range] RETURNS [Range] = INLINE {
Returns intersection of r1 and r2
RETURN [[x1: MAX[r1.x1, r2.x1], y1: MAX[r1.y1, r2.y1], x2: MIN[r1.x2, r2.x2], y2: MIN[r1.y2, r2.y2]]]
};
NonEmpty: PROC [r: Range] RETURNS [BOOL] = INLINE {
Returns r is not PipalInt.emptyRectangle [contains more points than the border]
RETURN [r.x1<r.x2 AND r.y1<r.y2]
};
InlineBBox: PROC [trans: PipalInt.Transformation, object: Pipal.Object] RETURNS [range: Range] = INLINE {
range ← RectToRange[PipalInt.BBox[object, trans]];
};
Divide: PROC [bucket: Range] RETURNS [bucket1, bucket2: Range] = INLINE {
bucket1 ← bucket2 ← bucket;
IF bucket.x2-bucket.x1>=bucket.y2-bucket.y1
THEN {mid: INT ← Halve[bucket.x1+bucket.x2]; bucket1.x2 ← mid; bucket2.x1 ← mid+1}
ELSE {mid: INT ← Halve[bucket.y1+bucket.y2]; bucket1.y2 ← mid; bucket2.y1 ← mid+1};
};
HashInstance: PROC [table: Table, trans: PipalInt.Transformation, object: Pipal.Object] RETURNS [hash: NAT] = INLINE {
hash ← HashRect[table, InlineBBox[trans, object]];
};
RectToRange: PROC [rect: PipalInt.Rectangle] RETURNS [Range] = INLINE {
RETURN [[x1: rect.base.x, y1: rect.base.y, x2: rect.base.x+rect.size.x, y2: rect.base.y+rect.size.y]];
};
HashRect: PROC [table: Table, range: Range] RETURNS [hash: NAT ← 1] = {
bucket: Range ← RectToRange[table.rect];
IF NOT Inside[range, bucket] THEN ERROR;
WHILE hash * 2 < table.data.size DO
bucket1, bucket2: Range;
[bucket1, bucket2] ← Divide[bucket];
SELECT TRUE FROM
Inside[range, bucket1] => {hash ← hash * 2;  bucket ← bucket1};
Inside[range, bucket2] => {hash ← hash * 2 + 1; bucket ← bucket2};
ENDCASE       => RETURN;
ENDLOOP;
};
Create: PUBLIC PROC [rect: PipalInt.Rectangle, logSize: NAT ← 2] RETURNS [Table] = {
data: REF TableData ← NEW [TableData[Basics.BITSHIFT[value: 1, count: logSize]]];
RETURN [NEW [TableRec ← [rect: rect, leafBuckets: 0, data: data]]];
};
ReHash: PROC [table: Table] = {
oldData: REF TableData ← table.data;
newData: REF TableData ← NEW [TableData[oldData.size*2]];
table.data ← newData;
table.leafBuckets ← 0;
FOR i: NAT IN [1 .. oldData.size) DO
values: LIST OF InstanceValue ← oldData[i];
WHILE values#NIL DO
hash: NAT ← HashInstance[table, values.first.trans, values.first.object];
IF newData[hash]=NIL THEN table.leafBuckets ← table.leafBuckets+1;
newData[hash] ← Cons[values.first, newData[hash]];
values ← Dispose[values];
ENDLOOP;
ENDLOOP;
};
Insert: PUBLIC ENTRY PROC [table: Table, trans: PipalInt.Transformation, object: Pipal.Object, value: Value] = {
ENABLE UNWIND => NULL;
hash: NAT ← HashInstance[table, trans, object];
IF table.data[hash]=NIL THEN table.leafBuckets ← table.leafBuckets+1;
table.data[hash] ← Cons[[trans, object, value], table.data[hash]];
IF table.leafBuckets * 2 > table.data.size THEN ReHash[table];
};
Enumerate: PUBLIC PROC [table: Table, action: PROC [PipalInt.Transformation, Pipal.Object, Value], rect: PipalInt.Rectangle ← PipalInt.fullRectangle] = {
EnumerateBucket: PROC [hash: NAT] = {
FOR ivs: LIST OF InstanceValue ← data[hash], ivs.rest WHILE ivs#NIL DO
IF Intersect[InlineBBox[ivs.first.trans, ivs.first.object], clipped]
THEN action[ivs.first.trans, ivs.first.object, ivs.first.value];
ENDLOOP;
};
data: REF TableData ← table.data; -- stored in the local frame to avoid concurrent inconsistencies
clipped: Range ← Intersection[RectToRange[rect], RectToRange[table.rect]];
hash1, hash2, hash: NAT;
IF NOT NonEmpty[clipped] THEN RETURN; -- PipalInt.emptyRectangle range
hash1 ← hash2 ← hash ← HashRect[table, clipped];
WHILE hash>0 DO EnumerateBucket[hash]; hash ← hash/2 ENDLOOP;
DO
hash1 ← 2 * hash1; hash2 ← 2 * hash2 + 1;
IF hash2>=data.size THEN RETURN;
FOR hash IN [hash1 .. hash2] DO EnumerateBucket[hash] ENDLOOP;
ENDLOOP;
};
DeleteOutside: PUBLIC ENTRY PROC [table: Table, rect: PipalInt.Rectangle ← PipalInt.emptyRectangle] = {
range: Range ← RectToRange[rect];
FOR i: NAT IN [1 .. table.data.size) DO
IF table.data[i]#NIL THEN {
current: LIST OF InstanceValue ← table.data[i];
previous: LIST OF InstanceValue ← NIL; -- either NIL or the node before current
allDeleted: BOOLTRUE;
UNTIL current = NIL DO
SELECT TRUE FROM
Intersect[range, InlineBBox[current.first.trans, current.first.object]] => {
allDeleted ← FALSE; previous ← current; current ← current.rest;
};
previous = NIL => current ← table.data[i] ← Dispose[current];
ENDCASE => current ← previous.rest ← Dispose[current];
ENDLOOP;
IF allDeleted THEN table.leafBuckets ← table.leafBuckets-1;
};
ENDLOOP;
};
For debugging!
Size: PROC [table: Table] RETURNS [size: INT ← 0] = {
Count: PROC [PipalInt.Transformation, Pipal.Object, Value] = {size ← size + 1};
Enumerate[table, Count];
};
END.