-- December 10, 1982 2:25 pm
-- program OldjunoSolverImpl.mesa, coded July 1981 by Greg Nelson

-- This is the solver for an interactive graphics program that allows a user to 
-- specify the positions of points by declaring geometrical contraints that
-- the positions are to satisfy. The program solves constraints (which can be
-- viewed as simultaneous polynomial equations) by an n-dimensional version of
-- Newton's method for finding the root of a differentiable function.
-- Last Edited by: Gnelson, October 11, 1983 7:00 pm

DIRECTORY  RealFns, JunoStorage, OldJunoSolver, JunoGraphics, Real, IO, JunoButtons;
           

OldJunoSolverImpl: PROGRAM
  IMPORTS  RealFns, JunoStorage, JunoGraphics, Real
  EXPORTS OldJunoSolver  = 

BEGIN OPEN JunoStorage, OldJunoSolver, JG: JunoGraphics;

displayingMotions: PUBLIC BOOL ← FALSE;
loggingPivots: BOOL ← FALSE;

rowmax: INTEGER = 100;

colmax: INTEGER = 100;

Tableau: TYPE = ARRAY[1..rowmax] OF ARRAY[1..colmax] OF REAL;

n,m: PUBLIC INT; -- the tableau a has n relevant rows and m relevant columns.

a: REF Tableau ← NEW[Tableau];

Tag: TYPE = {x, y, eq, geq};

TableauLabel: TYPE = RECORD [point: PointPtr, tag: Tag];

  -- each rows and column of the matrix represents a single scalar variable.
  -- If the row or column is labelled with the TableauLabel (p, x),
  -- then the scalar variable that the row or column represents is
  -- the x coordinate of the point p; similarly (p, y) means the y coordinate.
  -- The label (*, eq) means the scalar variable is a "slack variable" associated
  -- with some constraint; the variable should have the value zero in any
  -- solution.  The label (*, geq) menas the scalar variable should have a
  -- value greater than or equal to zero in any solution.   

rowhead: ARRAY[1..rowmax] OF TableauLabel;

colhead: ARRAY[1..colmax] OF TableauLabel;

TooManyConstraints: SIGNAL = CODE;

TooManyConstrainedPoints: SIGNAL = CODE;

Square: PROCEDURE[x:REAL] RETURNS [REAL] =  BEGIN RETURN [x*x] END;

Sqrt: PROCEDURE[x:REAL] RETURNS [REAL] =  BEGIN RETURN [RealFns.SqRt[x]] END;

Solve: PUBLIC PROC[epsilon: REAL] RETURNS [success: BOOLEAN] =
  {n: INT ← 0;
   SavePositions[];
   [] ← NewtonRaphson[];
   WHILE n < 20 AND SomePointMoved[] DO
     IF displayingMotions THEN DisplayMotions[]; 
     n ← n + 1;
     SavePositions[];
     [] ← NewtonRaphson[]
   ENDLOOP;
   RETURN [n < 20]}; 

SavePositions: PROC =
 {p: PointPtr ← pointLpad.link; 
  UNTIL p = pointRpad DO p.oldx ← Real.Fix[p.x]; p.oldy ← Real.Fix[p.y]; p ← p.link ENDLOOP};

SomePointMoved: PROC RETURNS [BOOL] =
  {p: PointPtr ← pointLpad.link;
   UNTIL p = pointRpad DO 
     IF p.oldx # Real.Fix[p.x] OR p.oldy # Real.Fix[p.y] THEN RETURN [TRUE];
     p ← p.link
   ENDLOOP;
   RETURN [FALSE]}; 
 
 DisplayMotions: PROC =
  {p: PointPtr ← pointLpad.link;
   UNTIL p = pointRpad DO 
     IF p.oldx # Real.Fix[p.x] OR p.oldy # Real.Fix[p.y]
        THEN JG.DrawEdge[p.oldx, p.oldy, p.x, p.y];
     p ← p.link
   ENDLOOP};
   
   
NewtonRaphson: PROC RETURNS [success: BOOLEAN] =
BEGIN
  
  NewtonStep: PROC[error: REAL,  gradient: TangentVector, tag: Tag] =
    -- processes one constraint whose current value is ERROR and whose current
    -- gradient is GRADIENT. Processes means adds row to matrix.
    BEGIN
      IF gradient = NIL THEN RETURN;
      n ← n + 1; -- add new row
      IF n > rowmax THEN SIGNAL TooManyConstraints[];
      rowhead[n] ← [point: NIL, tag: tag];
      FOR j: INT IN [2..m] DO a[n][j] ← 0.0 ENDLOOP;
      a[n][1] ← - error;
      UNTIL gradient = NIL DO
        IF ABS[gradient.x] > .000001 AND (NOT gradient.head.fixed) 
                                          AND (NOT gradient.head.tempfixed) THEN
	  {IF gradient.head.xheader = 0 
	   THEN -- add new column: 
	     {m ← m + 1;
	      IF m > colmax THEN SIGNAL TooManyConstrainedPoints[];
	      FOR i: INT IN [1..n] DO a[i][m] ← 0.0 ENDLOOP;
	      gradient.head.xheader ← m;
	      colhead[m] ← [point:gradient.head, tag: x]};
	   a[n][gradient.head.xheader] ← a[n][gradient.head.xheader] + gradient.x};
	   
	 IF ABS[gradient.y] > .000001 AND (NOT gradient.head.fixed) 
	                                   AND (NOT gradient.head.tempfixed) THEN
	  {IF gradient.head.yheader = 0 
	   THEN -- add new column: 
	     {m ← m + 1;
	      IF m > colmax THEN SIGNAL TooManyConstrainedPoints[];
	      FOR i: INT IN [1..n] DO a[i][m] ← 0.0 ENDLOOP;
	      gradient.head.yheader ← m;
	      colhead[m] ← [point:gradient.head, tag: y]};
	   a[n][gradient.head.yheader] ← a[n][gradient.head.yheader] + gradient.y};
	 
	 gradient ← gradient.tail;
       ENDLOOP;
     END;
     
 ConsTv: PROC[head:PointPtr, x:REAL, y:REAL, tail:TangentVector] RETURNS [TangentVector] =
  {tv:TangentVector ← NewTv[];
   tv.head ← head;
   tv.x ← x;
   tv.y ← y;
   tv.tail ← tail;
   RETURN [tv]};
     
 GradHor: PROC[p1, p2: PointPtr] RETURNS [tv:TangentVector] =
  -- return gradient of function p1.y - p2.y:
  {tv ← ConsTv[head:p1, x:0, y:1, tail:ConsTv[head:p2, x:0, y:-1, tail:NIL]]};

GradVer: PROC[p1, p2: PointPtr] RETURNS [tv:TangentVector] =
  -- return gradient of function p1.x - p2.x:
  {tv ← ConsTv[p1, 1, 0, ConsTv[p2, -1, 0, NIL]]};

GradCong: PROC[p1, p2, p3, p4: PointPtr] RETURNS [tv:TangentVector] =
  {tv ← ConsTv[p1, p1.x - p2.x, p1.y - p2.y,
          ConsTv[p2, p2.x - p1.x, p2.y - p1.y,
	    ConsTv[p3, p4.x - p3.x, p4.y - p3.y,
	      ConsTv[p4, p3.x - p4.x, p3.y - p4.y, NIL]]]]};

GradLin: PROC[pi, pj, pk, pl: PointPtr] RETURNS [tv: TangentVector] =
  {tv ← ConsTv[pi, pl.y - pk.y, pk.x - pl.x,
          ConsTv[pj, pk.y - pl.y, pl.x - pk.x,
	      ConsTv[pk, pi.y - pj.y, pj.x - pi.x, 
	      ConsTv[pl,  pj.y - pi.y, pi.x - pj.x, NIL]]]]};

GradCC: PROC[p1, p2, p3: PointPtr] RETURNS [tv: TangentVector] =
  {tv ← ConsTv[p1, p3.y - p2.y, p2.x-p3.x,
          ConsTv[p2, p1.y-p3.y, p3.x-p1.x,
	    ConsTv[p3, p2.y-p1.y,p1.x-p2.x, NIL]]]};
	
  p: PointPtr ← pointLpad.link;
  UNTIL p = pointRpad DO p.xheader ← 0; p.yheader ← 0; p ← p.link ENDLOOP;
   
  success ← TRUE;
  n ← 0;
  m ← 1; -- matrix has n rows and m columns.
  -- column 1 is the constant column.
 
  BEGIN -- handle counter-clockwise constraints:
    p: CCPtr ← ccLpad.link;
    UNTIL p = ccRpad DO
      NewtonStep[(p.j.y - p.i.y) * (p.k.x - p.i.x) - (p.j.x - p.i.x) * (p.k.y - p.i.y) + 500.0,
                 GradCC[p.i, p.j, p.k], geq];
      p ← p.link;
    ENDLOOP;
  END;

    BEGIN  -- handle string constraints:
    p: StringPtr ← stringLpad.link;
    UNTIL p = stringRpad DO
      NewtonStep [p.b4.x - p.b3.x - p.width, GradVer[p.b4, p.b3], eq];
      NewtonStep [p.b4.y - p.b3.y, GradHor[p.b4, p.b3], eq];
      p ← p.link;
    ENDLOOP;
  END;
 
  BEGIN  -- handle horizontal constraints:
    p: HorPtr ← horLpad.link;
    UNTIL  p = horRpad DO
      NewtonStep[p.i.y - p.j.y, GradHor[p.i, p.j], eq];
      p ← p.link
    ENDLOOP;
  END;

  BEGIN  -- handle vertical constraints:
    p: VerPtr ← verLpad.link;
    UNTIL  p = verRpad DO
      NewtonStep[p.i.x - p.j.x, GradVer[p.i, p.j], eq];
      p ← p.link;
    ENDLOOP;
  END;

  BEGIN -- handle parallel constraints:
    p: LinPtr ← lineLpad.link;
    UNTIL p = lineRpad DO
      NewtonStep[(p.j.x - p.i.x)*(p.k.y - p.l.y) + (p.l.x - p.k.x)*(p.j.y - p.i.y),
                 GradLin[p.i, p.j, p.k, p.l], eq];
      p ← p.link;
    ENDLOOP;
  END;

  BEGIN -- handle congruence constraints:
    p: CongPtr ← congLpad.link;
    UNTIL p = congRpad DO
      NewtonStep[0.5 * (Square[p.j.x - p.i.x] + Square[p.j.y - p.i.y]
                          - Square[p.l.x - p.k.x] - Square[p.l.y - p.k.y]),
	         GradCong[p.i, p.j, p.k, p.l], eq];
      p ← p.link;
    ENDLOOP;
  END;
   
  -- Now invert matrix, and move points:
  
   BEGIN
    i, j, k, l: INT;
    pp: REAL;
    -- k is the row in which we are pivoting.
    -- l is the column in which we are pivoting.
    -- i and j are miscellaneous row and column indices respectively
    -- p is the reciprocal of the pivot element; also used as temp for swapping.
    
    FOR k IN [1..n] DO
      IF rowhead[k].tag = eq OR rowhead[k].tag = geq AND a[k][1] < -.01
      THEN {-- set l so a[k,l] is largest of a[k,1], a[k, 2], a[k, 3], excluding
               -- columns in which we have already pivoted.
               pp ← 0;
               l ← 0;
               FOR j IN [2 .. m] 
                 DO IF ABS[a[k][j]] >= pp AND colhead[j].tag IN [x .. y] 
	              THEN {l ← j; pp ← ABS[a[k][l]]}
                 ENDLOOP;
               -- We will pivot at a[k,l], if it exists and is not too small, and is legal.
               -- First check that it is not too small:
               IF l = 0 OR ABS[a[k][l]] < .001 THEN
                 IF ABS[a[k][1]] > .01
	           -- then this row is an unsatisfiable constraint so: 
	           THEN {LOOP}
	           ELSE LOOP; -- row is identically zero; constraint redundant.
	           -- Next we must check that pivoting at a[k, l] will not ruin any
	           -- inequalities that are currently satisfied.  If there are any such inequalities,
	           -- we change k to the index of one of them (namely, the first that would become
	           -- unsatisfied), and pivot there instead.  That is the function of the following block:
	  
	           { -- adjust k to account for inequalities:
	             kk: INTEGER ← k; -- save k 
	             FOR i: INT IN [1..n] DO 
	                IF i # kk AND rowhead[i].tag = geq AND a[i][1] >= 0.0
	                           AND a[i][l] < 0.0 AND a[i][1] - a[i][l] * a[k][1] / a[k][l] < 0.0
	                THEN k ← i
	              ENDLOOP};
	       
               {temp:TableauLabel ← colhead[l]; colhead[l] ← rowhead[k]; rowhead[k] ← temp}; 
               pp ← 1.0 / a[k][l]; a[k][l] ← 1.0;
               -- divide everything in pivot row by the pivot element:
               FOR j IN [1..m] DO a[k][j] ← a[k][j] * pp ENDLOOP;
     
               FOR i IN [1..n] DO 
                          IF i # k THEN 
	           FOR j IN [1..m] DO 
	             IF j # l THEN -- for each a[i,j] outside the pivot row and column
	               a[i][j] ← a[i][j] - a[i][l] * a[k][j]; -- note that a[k,j] was already * pp.
               ENDLOOP ENDLOOP;
      
               -- Finally process pivot column:
               FOR i IN [1..n] DO IF i # k THEN a[i][l] ← -a[i][l] * pp ENDLOOP;
  	          }     

    ENDLOOP;
    
       -- now move the points:
       
    FOR i IN [1..n] DO
      IF rowhead[i].tag IN [x .. y]  
       -- as it will be except for redundant rows
       -- (or unsatisfiable rows)
       THEN
         IF rowhead[i].tag = x
           THEN rowhead[i].point.x ← rowhead[i].point.x + a[i][1]
	   ELSE rowhead[i].point.y ← rowhead[i].point.y + a[i][1];
    ENDLOOP;
                               
      END;
END; 
 
   
END.