-- JaMMath.mesa
-- Written by: John Warnock, February 27, 1979
-- Last changed by Doug Wyatt, January 12, 1981 5:42 PM

DIRECTORY
JaMMathDefs,
JaMMasterDefs USING [Frame, Object],
JaMControlDefs USING [GetCurrentFrame, NotifyCommand, NotifyStringObject],
JaMExecDefs USING [JaMError],
JaMFnsDefs USING [
GetReal, PopBoolean, PopInteger, PushBoolean, PushInteger, PushReal],
JaMStackDefs USING [Pop, Push],
JaMStringDefs USING [StringCompare],
InlineDefs USING [BITAND, BITNOT, BITOR, BITSHIFT, BITXOR],
Real USING [Fix, RealException],
RealFns USING [ArcTanDeg, CosDeg, Log, Power, SinDeg];

JaMMath: PROGRAM
IMPORTS JaMFnsDefs,JaMControlDefs,JaMStringDefs,JaMExecDefs,JaMStackDefs,
RealFns,Real,InlineDefs
EXPORTS JaMMathDefs =
BEGIN OPEN JaMStackDefs,JaMMasterDefs;

TypeError: PROCEDURE =
BEGIN ERROR JaMExecDefs.JaMError[TypeChk,TRUE] END;
OverFlwChk: PUBLIC PROCEDURE =
BEGIN ERROR JaMExecDefs.JaMError[OverFlow,TRUE] END;


MixedMode
: PUBLIC PROCEDURE[ob1,ob2: POINTER TO Object,
PInt: PROCEDURE[i,j: INTEGER, o: POINTER TO Object],
PLongInt: PROCEDURE[i,j: LONG INTEGER, o: POINTER TO Object],
PReal: PROCEDURE[i,j: REAL, o: POINTER TO Object]]
RETURNS[rs: Object] = {
ENABLE Real.RealException => ERROR JaMExecDefs.JaMError[OverFlow,TRUE];
rsptr: POINTER TO Object←@rs;
WITH dob1: ob1↑ SELECT FROM
IntegerType => WITH dob2: ob2↑ SELECT FROM
IntegerType =>
PInt[dob1.IntegerVal,dob2.IntegerVal, rsptr];
LongIntegerType=>
BEGIN
li: LONG INTEGER←dob1.IntegerVal;
PLongInt[li,dob2.LongIntegerVal, rsptr];
END;
RealType =>
BEGIN
r: REAL←dob1.IntegerVal;
PReal[r,dob2.RealVal, rsptr];
END;
ENDCASE => TypeError[];
LongIntegerType=> WITH dob2: ob2↑ SELECT FROM
IntegerType =>
BEGIN
li: LONG INTEGER←dob2.IntegerVal;
PLongInt[dob1.LongIntegerVal,li, rsptr];
END;
LongIntegerType=>
PLongInt[dob1.LongIntegerVal,dob2.LongIntegerVal, rsptr];
RealType =>
BEGIN
r: REAL←dob1.LongIntegerVal;
PReal[r,dob2.RealVal, rsptr];
END;
ENDCASE => TypeError[];
RealType => WITH dob2: ob2↑ SELECT FROM
IntegerType =>
BEGIN
r: REAL←dob2.IntegerVal;
PReal[dob1.RealVal,r, rsptr];
END;
LongIntegerType=>
BEGIN
r: REAL←dob2.LongIntegerVal;
PReal[dob1.RealVal,r, rsptr];
END;
RealType =>
PReal[dob1.RealVal,dob2.RealVal, rsptr];
ENDCASE => TypeError[];
ENDCASE => TypeError[];
};

LOverflowChk
: PUBLIC PROCEDURE[i,j: LONG INTEGER] RETURNS[BOOLEAN] =
BEGIN
--RETURN[IF((i<0)=(j<0)) THEN IF((i+j)<0)=(i<0) THEN FALSE ELSE TRUE ELSE FALSE];
RETURN[(i<0)=(j<0) AND ((i+j)<0)#(i<0)];
END;


Add: PUBLIC PROCEDURE =
BEGIN
frame: Frame←JaMControlDefs.GetCurrentFrame[];
ob2: Object←Pop[frame.opstk];
ob1: Object←Pop[frame.opstk];
Push[MixedMode[@ob1,@ob2,AddI,AddLI,AddR],frame.opstk];
END;

AddI: PROCEDURE[i,j: INTEGER, o: POINTER TO Object] =
BEGIN
l: LONG INTEGER←LONG[i] + LONG[j];
IF l > 32767 OR l < -LONG[32768] THEN o↑←[lit,LongIntegerType[l]]
ELSE o↑←[lit,IntegerType[i+j]];
END;

AddLI: PROCEDURE[i,j: LONG INTEGER, o: POINTER TO Object] =
BEGIN
IF LOverflowChk[i,j] THEN
BEGIN
ri: REAL←i;
rj: REAL←j;
o↑←[lit,RealType[ri+rj]];
END
ELSE o↑←[lit,LongIntegerType[i+j]];
END;

AddR: PROCEDURE[i,j: REAL, o: POINTER TO Object] =
BEGIN
ri: REAL←i;
rj: REAL←j;
o↑←[lit,RealType[ri+rj]];
END;



Sub
: PUBLIC PROCEDURE =
BEGIN
frame: Frame←JaMControlDefs.GetCurrentFrame[];
ob2: Object←Pop[frame.opstk];
ob1: Object←Pop[frame.opstk];
Push[MixedMode[@ob1,@ob2,SubI,SubLI,SubR],frame.opstk];
END;

SubI: PROCEDURE[i,j: INTEGER, o: POINTER TO Object] =
BEGIN
l: LONG INTEGER←LONG[i] - LONG[j];
IF l > 32767 OR l < -LONG[32768] THEN o↑←[lit,LongIntegerType[l]]
ELSE o↑←[lit,IntegerType[i-j]];
END;

SubLI: PROCEDURE[i,j: LONG INTEGER, o: POINTER TO Object] =
BEGIN
IF LOverflowChk[i,-j] THEN
BEGIN
ri: REAL←i;
rj: REAL←j;
o↑←[lit,RealType[ri-rj]];
END
ELSE o↑←[lit,LongIntegerType[i-j]];
END;

SubR: PROCEDURE[i,j: REAL, o: POINTER TO Object] =
BEGIN
ri: REAL←i;
rj: REAL←j;
o↑←[lit,RealType[ri-rj]];
END;


Mul
: PUBLIC PROCEDURE =
BEGIN
frame: Frame←JaMControlDefs.GetCurrentFrame[];
ob2: Object←Pop[frame.opstk];
ob1: Object←Pop[frame.opstk];
Push[MixedMode[@ob1,@ob2,MulI,MulLI,MulR],frame.opstk];
END;

MulI: PROCEDURE[i,j: INTEGER, o: POINTER TO Object] =
BEGIN
l: LONG INTEGER←LONG[i] * LONG[j];
IF l > 32767 OR l < -LONG[32768] THEN o↑←[lit,LongIntegerType[l]]
ELSE o↑←[lit,IntegerType[i*j]];
END;

MulLI: PROCEDURE[i,j: LONG INTEGER, o: POINTER TO Object] =
BEGIN
ri: REAL←i;
rj: REAL←j;
r: REAL←ri * rj;
IF r > 17777777777B OR r < 20000000000B THEN o↑←[lit,RealType[r]]
ELSE o↑←[lit,LongIntegerType[Real.Fix[r]]];
END;

MulR: PROCEDURE[i,j: REAL, o: POINTER TO Object] =
BEGIN
o↑←[lit,RealType[i*j]];
END;


Div
: PUBLIC PROCEDURE =
BEGIN
frame: Frame←JaMControlDefs.GetCurrentFrame[];
ob2: Object←Pop[frame.opstk];
ob1: Object←Pop[frame.opstk];
Push[MixedMode[@ob1,@ob2,DivI,DivLI,DivR],frame.opstk];
END;

DivI: PROCEDURE[i,j: INTEGER, o: POINTER TO Object] =
BEGIN
IF j = 0 THEN OverFlwChk[];
o↑←[lit,IntegerType[i/j]];
END;

DivLI: PROCEDURE[i,j: LONG INTEGER, o: POINTER TO Object] =
BEGIN
IF j = 0 THEN OverFlwChk[];
o↑←[lit,LongIntegerType[i/j]];
END;

DivR: PROCEDURE[i,j: REAL, o: POINTER TO Object] =
BEGIN
IF j = 0 THEN OverFlwChk[];
o↑←[lit,RealType[i/j]];
END;

Neg
: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
frame: Frame←JaMControlDefs.GetCurrentFrame[];
ob: Object←Pop[frame.opstk];
WITH dob: ob SELECT FROM
IntegerType =>
BEGIN
i: INTEGER←dob.IntegerVal;
PushInteger[-i];
END;
LongIntegerType =>
BEGIN
l: LONG INTEGER←dob.LongIntegerVal;
PushReal[-l];
END;
RealType =>
BEGIN
r: REAL←dob.RealVal;
PushReal[-r];
END;
ENDCASE => TypeError[];
END;

Sin
: PUBLIC PROCEDURE =
BEGIN
r: REAL←JaMFnsDefs.GetReal[];
JaMFnsDefs.PushReal[RealFns.SinDeg[r]];
END;

Cos
: PUBLIC PROCEDURE =
BEGIN
r: REAL←JaMFnsDefs.GetReal[];
JaMFnsDefs.PushReal[RealFns.CosDeg[r]];
END;

ATan
: PUBLIC PROCEDURE =
BEGIN
r1: REAL←JaMFnsDefs.GetReal[];
r2: REAL←JaMFnsDefs.GetReal[];
JaMFnsDefs.PushReal[RealFns.ArcTanDeg[r2,r1]];
END;

Exp
: PUBLIC PROCEDURE =
BEGIN
r1: REAL←JaMFnsDefs.GetReal[];
r2: REAL←JaMFnsDefs.GetReal[];
JaMFnsDefs.PushReal[RealFns.Power[r2,r1]];
END;

Log
: PUBLIC PROCEDURE =
BEGIN
r1: REAL←JaMFnsDefs.GetReal[];
r2: REAL←JaMFnsDefs.GetReal[];
JaMFnsDefs.PushReal[RealFns.Log[r2,r1]];
END;

Equal
: PUBLIC PROCEDURE =
BEGIN
frame: Frame←JaMControlDefs.GetCurrentFrame[];
ob2: Object←Pop[frame.opstk];
ob1: Object←Pop[frame.opstk];
WITH sob1: ob1 SELECT FROM
StringType => WITH sob2: ob2 SELECT FROM
StringType =>
BEGIN
eq: BOOLEAN←JaMStringDefs.StringCompare[sob1,sob2] =equal;
JaMFnsDefs.PushBoolean[eq];
END;
ENDCASE => TypeError[];
ENDCASE=> Push[MixedMode[@ob1,@ob2,EqI,EqLI,EqR],frame.opstk];
END;

EqI: PROCEDURE[i,j: INTEGER, o: POINTER TO Object] =
BEGIN
o↑←[lit,BooleanType[BooleanVal:i = j]];
END;

EqLI: PROCEDURE[i,j: LONG INTEGER, o: POINTER TO Object] =
BEGIN
o↑←[lit,BooleanType[BooleanVal:i = j]];
END;

EqR: PROCEDURE[i,j: REAL, o: POINTER TO Object] =
BEGIN
o↑←[lit,BooleanType[BooleanVal:i = j]];
END;


LessThan: PUBLIC PROCEDURE =
BEGIN
frame: Frame←JaMControlDefs.GetCurrentFrame[];
ob2: Object←Pop[frame.opstk];
ob1: Object←Pop[frame.opstk];
WITH sob1: ob1 SELECT FROM
StringType => WITH sob2: ob2 SELECT FROM
StringType =>
BEGIN
ls: BOOLEAN←JaMStringDefs.StringCompare[sob1,sob2]=less;
JaMFnsDefs.PushBoolean[ls];
END;
ENDCASE => TypeError[];
ENDCASE=> Push[MixedMode[@ob1,@ob2,LtI,LtLI,LtR],frame.opstk];
END;

LtI: PROCEDURE[i,j: INTEGER, o: POINTER TO Object] =
BEGIN
o↑←[lit,BooleanType[BooleanVal:i < j]];
END;

LtLI: PROCEDURE[i,j: LONG INTEGER, o: POINTER TO Object] =
BEGIN
o↑←[lit,BooleanType[BooleanVal:i < j]];
END;

LtR: PROCEDURE[i,j: REAL, o: POINTER TO Object] =
BEGIN
o↑←[lit,BooleanType[BooleanVal:i < j]];
END;


GreaterThan: PUBLIC PROCEDURE =
BEGIN
frame: Frame←JaMControlDefs.GetCurrentFrame[];
ob2: Object←Pop[frame.opstk];
ob1: Object←Pop[frame.opstk];
WITH sob1: ob1 SELECT FROM
StringType => WITH sob2: ob2 SELECT FROM
StringType =>
BEGIN
gr: BOOLEAN←JaMStringDefs.StringCompare[sob1,sob2]=greater;
JaMFnsDefs.PushBoolean[gr];
END;
ENDCASE => TypeError[];
ENDCASE=> Push[MixedMode[@ob1,@ob2,GtI,GtLI,GtR],frame.opstk];
END;

GtI: PROCEDURE[i,j: INTEGER, o: POINTER TO Object] =
BEGIN
o↑←[lit,BooleanType[BooleanVal:i > j]];
END;

GtLI: PROCEDURE[i,j: LONG INTEGER, o: POINTER TO Object] =
BEGIN
o↑←[lit,BooleanType[BooleanVal:i > j]];
END;

GtR: PROCEDURE[i,j: REAL, o: POINTER TO Object] =
BEGIN
o↑←[lit,BooleanType[BooleanVal:i > j]];
END;


Not: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
PushBoolean[NOT PopBoolean[]];
END;

And: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
b1: BOOLEAN←PopBoolean[];
b2: BOOLEAN←PopBoolean[];
PushBoolean[b2 AND b1];
END;

Or: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
b1: BOOLEAN←PopBoolean[];
b2: BOOLEAN←PopBoolean[];
PushBoolean[b2 OR b1];
END;

Xor: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
b1: BOOLEAN←PopBoolean[];
b2: BOOLEAN←PopBoolean[];
PushBoolean[b1 # b2];
END;

BitNot: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
PushInteger[InlineDefs.BITNOT[PopInteger[]]];
END;

BitAnd: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
i1: INTEGER = PopInteger[];
i2: INTEGER = PopInteger[];
PushInteger[InlineDefs.BITAND[i1,i2]];
END;

BitOr: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
i1: INTEGER = PopInteger[];
i2: INTEGER = PopInteger[];
PushInteger[InlineDefs.BITOR[i1,i2]];
END;

BitXor: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
i1: INTEGER = PopInteger[];
i2: INTEGER = PopInteger[];
PushInteger[InlineDefs.BITXOR[i1,i2]];
END;

BitShift: PUBLIC PROCEDURE =
BEGIN OPEN JaMFnsDefs;
shift: INTEGER = PopInteger[];
word: INTEGER = PopInteger[];
PushInteger[InlineDefs.BITSHIFT[word,shift]];
END;

TypeChk: StringType Object;
OverFlow: StringType Object;

StartMath: PROCEDURE =
BEGIN OPEN JaMControlDefs;
NotifyStringObject[@TypeChk, ".typechk"L];
NotifyStringObject[@OverFlow, ".overflow"L];

-- Math commands
NotifyCommand[".add"L,Add];
NotifyCommand[".sub"L,Sub];
NotifyCommand[".mul"L,Mul];
NotifyCommand[".div"L,Div];
NotifyCommand[".neg"L,Neg];
NotifyCommand[".sin"L,Sin];
NotifyCommand[".cos"L,Cos];
NotifyCommand[".atan"L,ATan];
NotifyCommand[".exp"L,Exp];
NotifyCommand[".log"L,Log];
NotifyCommand[".eq"L,Equal];
NotifyCommand[".lt"L,LessThan];
NotifyCommand[".gt"L,GreaterThan];
NotifyCommand[".not"L,Not];
NotifyCommand[".and"L,And];
NotifyCommand[".or"L,Or];
NotifyCommand[".xor"L,Xor];
NotifyCommand[".bitnot"L,BitNot];
NotifyCommand[".bitand"L,BitAnd];
NotifyCommand[".bitor"L,BitOr];
NotifyCommand[".bitxor"L,BitXor];
NotifyCommand[".bitshift"L,BitShift];
END;

-- Initialization
StartMath;

END.

DKW March 28, 1980 4:55 PM
added StartMath

DKW April 1, 1980 3:48 PM
now uses NotifyCommand, NotifyStringObject

DKW May 31, 1980 1:05 AM
updated for Mesa6

DKW July 14, 1980 11:19 PM
FloatFns => RealFns

DKW January 12, 1981 5:42 PM
MixedMode catches RealException