aboutsummaryrefslogtreecommitdiff
path: root/src/interp/i-resolv.boot
diff options
context:
space:
mode:
Diffstat (limited to 'src/interp/i-resolv.boot')
-rw-r--r--src/interp/i-resolv.boot800
1 files changed, 800 insertions, 0 deletions
diff --git a/src/interp/i-resolv.boot b/src/interp/i-resolv.boot
new file mode 100644
index 00000000..ec359b1c
--- /dev/null
+++ b/src/interp/i-resolv.boot
@@ -0,0 +1,800 @@
+-- Copyright (c) 1991-2002, The Numerical ALgorithms Group Ltd.
+-- All rights reserved.
+--
+-- Redistribution and use in source and binary forms, with or without
+-- modification, are permitted provided that the following conditions are
+-- met:
+--
+-- - Redistributions of source code must retain the above copyright
+-- notice, this list of conditions and the following disclaimer.
+--
+-- - Redistributions in binary form must reproduce the above copyright
+-- notice, this list of conditions and the following disclaimer in
+-- the documentation and/or other materials provided with the
+-- distribution.
+--
+-- - Neither the name of The Numerical ALgorithms Group Ltd. nor the
+-- names of its contributors may be used to endorse or promote products
+-- derived from this software without specific prior written permission.
+--
+-- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
+-- IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
+-- TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
+-- PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
+-- OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+-- EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+-- PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+-- PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+-- LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+-- NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+-- SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+
+import '"i-object"
+)package "BOOT"
+
+resolveTypeList u ==
+ u is [a,:tail] =>
+
+ -- if the list consists entirely of variables then keep it explicit
+ allVars :=
+ a is ['Variable,v] => [v]
+ nil
+ while allVars for b in tail repeat
+ allVars :=
+ b is ['Variable,v] => insert(v, allVars)
+ nil
+ allVars =>
+ null rest allVars => ['Variable, first allVars]
+ ['OrderedVariableList,nreverse allVars]
+
+ for md in tail repeat
+ a := resolveTT(md,a)
+ null a => return nil
+ a
+ throwKeyedMsg("S2IR0002",NIL)
+
+-- resolveTT is in CLAMMED BOOT
+
+resolveTypeListAny tl ==
+ rt := resolveTypeList tl
+ null rt => $Any
+ rt
+
+resolveTTAny(t1,t2) ==
+ (t3 := resolveTT(t1, t2)) => t3
+ $Any
+
+resolveTT1(t1,t2) ==
+ -- this is the main symmetric resolve
+ -- first it looks for equal constructors on both sides
+ -- then it tries to use a rewrite rule
+ -- and finally it builds up a tower
+ t1=t2 => t1
+ (t1 = '$NoValueMode) or (t2 = '$NoValueMode) => NIL
+ (t1 = $Void) or (t2 = $Void) => $Void
+ (t1 = $Any) or (t2 = $Any) => $Any
+ t1 = '(Exit) => t2
+ t2 = '(Exit) => t1
+ t1 is ['Union,:.] => resolveTTUnion(t1,t2)
+ t2 is ['Union,:.] => resolveTTUnion(t2,t1)
+ STRINGP(t1) =>
+ t2 = $String => t2
+ NIL
+ STRINGP(t2) =>
+ t1 = $String => t1
+ NIL
+ null acceptableTypesToResolve(t1,t2) => NIL
+ if compareTT(t1,t2) then
+ t := t1
+ t1 := t2
+ t2 := t
+ (t := resolveTTSpecial(t1,t2)) and isValidType t => t
+ (t := resolveTTSpecial(t2,t1)) and isValidType t => t
+ isSubTowerOf(t1,t2) and canCoerceFrom(t1,t2) => t2
+ isSubTowerOf(t2,t1) and canCoerceFrom(t2,t1) => t1
+ t := resolveTTRed(t1,t2) => t
+ t := resolveTTCC(t1,t2) => t
+ (t := resolveTTEq(t1,t2)) and isValidType t => t
+ [c1,:arg1] := deconstructT t1
+ arg1 and
+ [c2,:arg2] := deconstructT t2
+ arg2 and
+ t := resolveTT1(last arg1,last arg2)
+ t and ( resolveTT2(c1,c2,arg1,arg2,t) or
+ resolveTT2(c2,c1,arg2,arg1,t) )
+
+acceptableTypesToResolve(t1,t2) ==
+ -- this is temporary. It ensures that two types that have coerces
+ -- that really should be converts don't automatically resolve.
+ -- when the coerces go away, so will this.
+ acceptableTypesToResolve1(t1,t2) and
+ acceptableTypesToResolve1(t2,t1)
+
+acceptableTypesToResolve1(t1,t2) ==
+ t1 = $Integer =>
+ t2 = $String => NIL
+ true
+ t1 = $DoubleFloat or t1 = $Float =>
+ t2 = $String => NIL
+ t2 = '(RationalNumber) => NIL
+ t2 = [$QuotientField, $Integer] => NIL
+ true
+ true
+
+resolveTT2(c1,c2,arg1,arg2,t) ==
+ -- builds a tower and tests for all the necessary coercions
+ t0 := constructM(c2,replaceLast(arg2,t))
+ canCoerceFrom(t,t0) and
+ t1 := constructM(c1,replaceLast(arg1,t0))
+ canCoerceFrom(t0,t1) and t1
+
+resolveTTUnion(t1 is ['Union,:doms],t2) ==
+ unionDoms1 :=
+ doms and first doms is [":",:.] =>
+ tagged := true
+ [t for [.,.,t] in doms]
+ tagged := false
+ doms
+ member(t2,unionDoms1) => t1
+ tagged => NIL
+ t2 isnt ['Union,:doms2] =>
+ ud := nil
+ bad := nil
+ for d in doms while ^bad repeat
+ d = '"failed" => ud := [d,:ud]
+ null (d' := resolveTT(d,t2)) => bad := true
+ ud := [d',:ud]
+ bad => NIL
+ ['Union,:REMDUP reverse ud]
+ ud := nil
+ bad := nil
+ for d in doms2 while ^bad repeat
+ d = '"failed" => ud := append(ud,[d])
+ null (d' := resolveTTUnion(t1,d)) => bad := true
+ ud := append(ud,CDR d')
+ bad => NIL
+ ['Union,:REMDUP ud]
+
+resolveTTSpecial(t1,t2) ==
+ -- tries to resolve things that would otherwise get mangled in the
+ -- rest of the resolve world. I'll leave it for Albi to fix those
+ -- things. (RSS 1/-86)
+
+ -- following is just an efficiency hack
+ (t1 = '(Symbol) or t1 is ['OrderedVariableList,.]) and PAIRP(t2) and
+ CAR(t2) in '(Polynomial RationalFunction) => t2
+
+ (t1 = '(Symbol)) and ofCategory(t2, '(IntegerNumberSystem)) =>
+ resolveTT1(['Polynomial, t2], t2)
+
+ t1 = '(AlgebraicNumber) and (t2 = $Float or t2 = $DoubleFloat) =>
+ ['Expression, t2]
+ t1 = '(AlgebraicNumber) and (t2 = ['Complex, $Float] or t2 = ['Complex, $DoubleFloat]) =>
+ ['Expression, CADR t2]
+
+ t1 = '(AlgebraicNumber) and t2 is ['Complex,.] =>
+ resolveTT1('(Expression (Integer)), t2)
+
+ t1 is ['SimpleAlgebraicExtension,F,Rep,poly] =>
+ t2 = Rep => t1
+ t2 is ['UnivariatePolynomial,x,R] and (t3 := resolveTT(t1, R)) =>
+ ['UnivariatePolynomial,x,t3]
+ t2 is ['Variable,x] and (t3 := resolveTT(t1, F)) =>
+ ['UnivariatePolynomial,x,t3]
+ t2 is ['Polynomial,R] and (R' := resolveTT(Rep, t2)) =>
+ R' = Rep => t1
+ ['Polynomial,t1]
+ canCoerceFrom(t2,F) => t1
+ nil
+ t1 = $PositiveInteger and ofCategory(t2,'(Ring)) =>
+ resolveTT1($Integer,t2)
+ t1 = $NonNegativeInteger and ofCategory(t2,'(Ring)) =>
+ resolveTT1($Integer,t2)
+ t1 is ['OrderedVariableList,[x]] => resolveTTSpecial(['Variable, x], t2)
+ t1 is ['OrderedVariableList,vl] =>
+ ofCategory(t2,'(Ring)) => resolveTT(['Polynomial,'(Integer)],t2)
+ resolveTT($Symbol,t2)
+ t1 is ['Variable,x] =>
+ EQCAR(t2,'SimpleAlgebraicExtension) => resolveTTSpecial(t2,t1)
+ t2 is ['UnivariatePolynomial,y,S] =>
+ x = y => t2
+ resolveTT1(['UnivariatePolynomial,x,'(Integer)],t2)
+ t2 is ['Variable,y] =>
+ x = y => t1
+-- ['OrderedVariableList, MSORT [x,y]]
+ $Symbol
+ t2 = '(Symbol) => t2
+ t2 is ['Polynomial,.] => t2
+ t2 is ['OrderedVariableList, vl] and member(x,vl) => t2
+ isPolynomialMode t2 => nil
+ ofCategory(t2, '(IntegerNumberSystem)) => resolveTT(['Polynomial, t2], t2)
+ resolveTT(['Polynomial,'(Integer)],t2)
+ t1 is ['FunctionCalled,f] and t2 is ['FunctionCalled,g] =>
+ null (mf := get(f,'mode,$e)) => NIL
+ null (mg := get(g,'mode,$e)) => NIL
+ mf ^= mg => NIL
+ mf
+ t1 is ['UnivariatePolynomial,x,S] =>
+ EQCAR(t2,'Variable) =>
+ resolveTTSpecial(t2,t1)
+ EQCAR(t2,'SimpleAlgebraicExtension) =>
+ resolveTTSpecial(t2,t1)
+ t2 is ['UnivariatePolynomial,y,T] =>
+ (x = y) and (U := resolveTT1(S,T)) and ['UnivariatePolynomial,x,U]
+ nil
+ t1 = '(Pi) =>
+ t2 is ['Complex,d] => defaultTargetFE t2
+ t2 is ['AlgebraicNumber] => defaultTargetFE t2
+ EQCAR(t2, 'Variable) or t2 = $Symbol =>
+ defaultTargetFE($Symbol)
+ t2 is ['Polynomial, .] or t2 is ['Fraction, ['Polynomial, .]] =>
+ defaultTargetFE(t2)
+ nil
+ t1 is ['Polynomial,['Complex,u1]] and t2 is ['Complex,u2] =>
+ resolveTT1(t1,u2)
+ t1 is ['Polynomial,R] and t2 is ['Complex,S] =>
+ containsPolynomial(S) => resolveTT1(['Polynomial,['Complex,R]],t2)
+ ['Polynomial,['Complex,resolveTT1(R,S)]]
+ t1 is ['Expression, R] and t2 is ['Complex,S] =>
+ dom' := resolveTT(R, t2)
+ null dom' => nil
+ ['Expression, dom']
+ t1 is ['Segment, dom] and t2 isnt ['Segment,.] =>
+ dom' := resolveTT(dom, t2)
+ null dom' => nil
+ ['Segment, dom']
+ nil
+
+resolveTTCC(t1,t2) ==
+ -- tries to use canCoerceFrom information to see if types can be
+ -- coerced to one another
+ gt21 := GGREATERP(t2,t1)
+ (c12 := canCoerceFrom(t1,t2)) and gt21 => t2
+ c21 := canCoerceFrom(t2,t1)
+ null (c12 or c21) => NIL
+ c12 and not c21 => t2
+ c21 and not c12 => t1
+ -- both are coerceable to each other
+ if gt21 then t1 else t2
+
+resolveTTEq(t1,t2) ==
+ -- tries to find the constructor of t1 somewhere in t2 (or vice versa)
+ -- and move the other guy to the top
+ [c1,:arg1] := deconstructT t1
+ [c2,:arg2] := deconstructT t2
+ t := resolveTTEq1(c1,arg1,[c2,arg2]) => t
+ t := ( arg1 and resolveTTEq2(c2,arg2,[c1,arg1]) ) => t
+ arg2 and resolveTTEq2(c1,arg1,[c2,arg2])
+
+resolveTTEq1(c1,arg1,TL is [c2,arg2,:.]) ==
+ -- takes care of basic types and of types with the same constructor
+ -- calls resolveTT1 on the arguments in the second case
+ null arg1 and null arg2 =>
+ canCoerceFrom(c1,c2) => constructTowerT(c2,CDDR TL)
+ canCoerceFrom(c2,c1) and constructTowerT(c1,CDDR TL)
+ c1=c2 and
+ [c2,arg2,:TL] := bubbleType TL
+ until null arg1 or null arg2 or not t repeat
+ t := resolveTT1(CAR arg1,CAR arg2) =>
+ arg := CONS(t,arg)
+ arg1 := CDR arg1
+ arg2 := CDR arg2
+ t and null arg1 and null arg2 and
+ t0 := constructM(c1,nreverse arg)
+ constructTowerT(t0,TL)
+
+resolveTTEq2(c1,arg1,TL is [c,arg,:.]) ==
+ -- tries to resolveTTEq the type [c1,arg1] with the last argument
+ -- of the type represented by TL
+ [c2,:arg2] := deconstructT last arg
+ TL := [c2,arg2,:TL]
+ t := resolveTTEq1(c1,arg1,TL) => t
+ arg2 and resolveTTEq2(c1,arg1,TL)
+
+resolveTTRed(t1,t2) ==
+ -- the same function as resolveTTEq, but instead of testing for
+ -- constructor equality, it looks whether a rewrite rule can be applied
+ t := resolveTTRed1(t1,t2,NIL) => t
+ [c1,:arg1] := deconstructT t1
+ t := arg1 and resolveTTRed2(t2,last arg1,[c1,arg1]) => t
+ [c2,:arg2] := deconstructT t2
+ arg2 and resolveTTRed2(t1,last arg2,[c2,arg2])
+
+resolveTTRed1(t1,t2,TL) ==
+ -- tries to apply a reduction rule on (Resolve t1 t2)
+ -- then it creates a type using the result and TL
+ EQ(t,term1RW(t := ['Resolve,t1,t2],$Res)) and
+ EQ(t,term1RW(t := ['Resolve,t2,t1],$Res)) => NIL
+ [c2,:arg2] := deconstructT t2
+ [c2,arg2,:TL] := bubbleType [c2,arg2,:TL]
+ t2 := constructM(c2,arg2)
+ l := term1RWall(['Resolve,t1,t2],$Res)
+ for t0 in l until t repeat t := resolveTTRed3 t0
+ l and t => constructTowerT(t,TL)
+ l := term1RWall(['Resolve,t2,t1],$Res)
+ for t0 in l until t repeat t := resolveTTRed3 t0
+ l and t and constructTowerT(t,TL)
+
+resolveTTRed2(t1,t2,TL) ==
+ -- tries to resolveTTRed t1 and t2 and build a type using TL
+ t := resolveTTRed1(t1,t2,TL) => t
+ [c2,:arg2] := deconstructT t2
+ arg2 and resolveTTRed2(t1,last arg2,[c2,arg2,:TL])
+
+resolveTTRed3(t) ==
+ -- recursive resolveTTRed which handles all subterms of the form
+ -- (Resolve t1 t2) or subterms which have to be interpreted
+ atom t => t
+ t is ['Resolve,a,b] =>
+ ( t1 := resolveTTRed3 a ) and ( t2 := resolveTTRed3 b ) and
+ resolveTT1(t1,t2)
+ t is ['Incl,a,b] => member(a,b) and b
+ t is ['SetDiff,a,b] => intersection(a,b) and SETDIFFERENCE(a,b)
+ t is ['SetComp,a,b] =>
+ and/[member(x,a) for x in b] and SETDIFFERENCE(a,b)
+ t is ['SetInter,a,b] => intersection(a,b)
+ t is ['SetUnion,a,b] => union(a,b)
+ t is ['VarEqual,a,b] => (a = b) and a
+ t is ['SetEqual,a,b] =>
+ (and/[member(x,a) for x in b] and "and"/[member(x,b) for x in a]) and a
+ [( atom x and x ) or ((not cs and x and not interpOp? x and x)
+ or resolveTTRed3 x) or return NIL
+ for x in t for cs in GETDATABASE(CAR t, 'COSIG) ]
+
+interpOp?(op) ==
+ PAIRP(op) and
+ CAR(op) in '(Incl SetDiff SetComp SetInter SetUnion VarEqual SetEqual)
+
+--% Resolve Type with Category
+
+resolveTCat(t,c) ==
+ -- this function attempts to find a type tc of category c such that
+ -- t can be coerced to tc. NIL returned for failure.
+ -- Example: t = Integer, c = Field ==> tc = RationalNumber
+
+ -- first check whether t already belongs to c
+ ofCategory(t,c) => t
+
+ -- if t is built by a parametrized constructor and there is a
+ -- condition on the parameter that matches the category, try to
+ -- recurse. An example of this is (G I, Field) -> G RN
+
+ rest(t) and (tc := resolveTCat1(t,c)) => tc
+
+ -- now check some specific niladic categories
+ c in '((Field) (EuclideanDomain)) and ofCategory(t,'(IntegralDomain))=>
+ eqType [$QuotientField, t]
+
+ c = '(Field) and t = $Symbol => ['RationalFunction,$Integer]
+
+ c = '(Ring) and t is ['FactoredForm,t0] => ['FactoredRing,t0]
+
+ (t is [t0]) and (sd := getImmediateSuperDomain(t0)) and sd ^= t0 =>
+ resolveTCat(sd,c)
+
+ SIZE(td := deconstructT t) ^= 2=> NIL
+ SIZE(tc := deconstructT c) ^= 2 => NIL
+ ut := underDomainOf t
+ null isValidType(uc := last tc) => NIL
+ null canCoerceFrom(ut,uc) => NIL
+ nt := constructT(first td,[uc])
+ ofCategory(nt,c) => nt
+ NIL
+
+resolveTCat1(t,c) ==
+ -- does the hard work of looking at conditions on under domains
+ -- if null (ut := getUnderModeOf(t)) then ut := last dt
+ null (conds := getConditionsForCategoryOnType(t,c)) => NIL
+--rest(conds) => NIL -- will handle later
+ cond := first conds
+ cond isnt [.,["has", pat, c1],:.] => NIL
+ rest(c1) => NIL -- make it simple
+
+ argN := 0
+ t1 := nil
+
+ for ut in rest t for i in 1.. while (argN = 0) repeat
+ sharp := INTERNL('"#",STRINGIMAGE i)
+ sharp = pat =>
+ argN := i
+ t1 := ut
+
+ null t1 => NIL
+ null (t1' := resolveTCat(t1,c1)) => NIL
+ t' := copy t
+ t'.argN := t1'
+ t'
+
+getConditionsForCategoryOnType(t,cat) ==
+ getConditionalCategoryOfType(t,[NIL],['ATTRIBUTE,cat])
+
+getConditionalCategoryOfType(t,conditions,match) ==
+ if PAIRP t then t := first t
+ t in '(Union Mapping Record) => NIL
+ conCat := GETDATABASE(t,'CONSTRUCTORCATEGORY)
+ REMDUP CDR getConditionalCategoryOfType1(conCat,conditions,match,[NIL])
+
+getConditionalCategoryOfType1(cat,conditions,match,seen) ==
+ cat is ['Join,:cs] or cat is ['CATEGORY,:cs] =>
+ null cs => conditions
+ getConditionalCategoryOfType1([first cat,:rest cs],
+ getConditionalCategoryOfType1(first cs,conditions,match,seen),
+ match,seen)
+ cat is ['IF,., cond,.] =>
+ matchUpToPatternVars(cond,match,NIL) =>
+ RPLACD(conditions,CONS(cat,CDR conditions))
+ conditions
+ conditions
+ cat is [catName,:.] and (GETDATABASE(catName,'CONSTRUCTORKIND) = 'category) =>
+ cat in CDR seen => conditions
+ RPLACD(seen,[cat,:CDR seen])
+ subCat := GETDATABASE(catName,'CONSTRUCTORCATEGORY)
+ -- substitute vars of cat into category
+ for v in rest cat for vv in $TriangleVariableList repeat
+ subCat := SUBST(v,vv,subCat)
+ getConditionalCategoryOfType1(subCat,conditions,match,seen)
+ conditions
+
+matchUpToPatternVars(pat,form,patAlist) ==
+ -- tries to match pattern variables (of the # form) in pat
+ -- against expressions in form. If one is found, it is checked
+ -- against the patAlist to make sure we are using the same expression
+ -- each time.
+ EQUAL(pat,form) => true
+ isSharpVarWithNum(pat) =>
+ -- see is pattern variable is in alist
+ (p := assoc(pat,patAlist)) => EQUAL(form,CDR p)
+ patAlist := [[pat,:form],:patAlist]
+ true
+ PAIRP(pat) =>
+ not (PAIRP form) => NIL
+ matchUpToPatternVars(CAR pat, CAR form,patAlist) and
+ matchUpToPatternVars(CDR pat, CDR form,patAlist)
+ NIL
+
+--% Resolve Type with Mode
+
+-- only implemented for nullary control-L's (which stand for types)
+
+resolveTMOrCroak(t,m) ==
+ resolveTM(t,m) or throwKeyedMsg("S2IR0004",[t,m])
+
+resolveTM(t,m) ==
+ -- resolves a type with a mode which may be partially specified
+ startTimingProcess 'resolve
+ $Subst : local := NIL
+ $Coerce : local := 'T
+ t := eqType t
+ m := eqType SUBSTQ("**",$EmptyMode,m)
+ tt := resolveTM1(t,m)
+ result := tt and isValidType tt and eqType tt
+ stopTimingProcess 'resolve
+ result
+
+resolveTM1(t,m) ==
+ -- general resolveTM, which looks for a term variable
+ -- otherwise it looks whether the type has the same top level
+ -- constructor as the mode, looks for a rewrite rule, or builds up
+ -- a tower
+ t=m => t
+ m is ['Union,:.] => resolveTMUnion(t,m)
+ m = '(Void) => m
+ m = '(Any) => m
+ m = '(Exit) => t
+ containsVars m =>
+ isPatternVar m =>
+ p := ASSQ(m,$Subst) =>
+ $Coerce =>
+ tt := resolveTT1(t,CDR p) => RPLACD(p,tt) and tt
+ NIL
+ t=CDR p and t
+ $Subst := CONS(CONS(m,t),$Subst)
+ t
+ atom(t) or atom(m) => NIL
+ (t is ['Record,:tr]) and (m is ['Record,:mr]) and
+ (tt := resolveTMRecord(tr,mr)) => tt
+ t is ['Record,:.] or m is ['Record,:.] => NIL
+ t is ['Variable, .] and m is ['Mapping, :.] => m
+ t is ['FunctionCalled, .] and m is ['Mapping, :.] => m
+ if isEqualOrSubDomain(t, $Integer) then
+ t := $Integer
+ tt := resolveTMEq(t,m) => tt
+ $Coerce and
+ tt := resolveTMRed(t,m) => tt
+ resolveTM2(t,m)
+ $Coerce and canCoerceFrom(t,m) and m
+
+resolveTMRecord(tr,mr) ==
+ #tr ^= #mr => NIL
+ ok := true
+ tt := NIL
+ for ta in tr for ma in mr while ok repeat
+ -- element is [':,tag,mode]
+ CADR(ta) ^= CADR(ma) => ok := NIL -- match tags
+ ra := resolveTM1(CADDR ta, CADDR ma) -- resolve modes
+ null ra => ok := NIL
+ tt := CONS([CAR ta,CADR ta,ra],tt)
+ null ok => NIL
+ ['Record,nreverse tt]
+
+resolveTMUnion(t, m is ['Union,:ums]) ==
+ isTaggedUnion m => resolveTMTaggedUnion(t,m)
+ -- resolves t with a Union type
+ t isnt ['Union,:uts] =>
+ ums := REMDUP spliceTypeListForEmptyMode([t],ums)
+ ums' := nil
+ success := nil
+ for um in ums repeat
+ (um' := resolveTM1(t,um)) =>
+ success := true
+ um' in '(T TRUE) => ums' := [um,:ums']
+ ums' := [um',:ums']
+ ums' := [um,:ums']
+ -- remove any duplicate domains that might have been created
+ m' := ['Union,:REMDUP reverse ums']
+ success =>
+ null CONTAINED('_*_*,m') => m'
+ t = $Integer => NIL
+ resolveTM1($Integer,m')
+ NIL
+ -- t is actually a Union if we got here
+ ums := REMDUP spliceTypeListForEmptyMode(uts,ums)
+ bad := nil
+ doms := nil
+ for ut in uts while ^bad repeat
+ (m' := resolveTMUnion(ut,['Union,:ums])) =>
+ doms := append(CDR m',doms)
+ bad := true
+ bad => NIL
+ ['Union,:REMDUP doms]
+
+resolveTMTaggedUnion(t, m is ['Union,:ums]) ==
+ NIL
+
+spliceTypeListForEmptyMode(tl,ml) ==
+ -- splice in tl for occurrence of ** in ml
+ null ml => nil
+ ml is [m,:ml'] =>
+ m = "**" => append(tl,spliceTypeListForEmptyMode(tl,ml'))
+ [m,:spliceTypeListForEmptyMode(tl,ml')]
+
+resolveTM2(t,m) ==
+ -- resolves t with the last argument of m and builds up a tower
+ [cm,:argm] := deconstructT m
+ argm and
+ tt := resolveTM1(t,last argm)
+ tt and
+ ttt := constructM(cm,replaceLast(argm,tt))
+ ttt and canCoerceFrom(tt,ttt) and ttt
+
+resolveTMEq(t,m) ==
+ -- tests whether t and m have the same top level constructor, which,
+ -- in the case of t, could be bubbled up
+ (res := resolveTMSpecial(t,m)) => res
+ [cm,:argm] := deconstructT m
+ c := containsVars cm
+ TL := NIL
+ until b or not t repeat
+ [ct,:argt] := deconstructT t
+ b :=
+ c =>
+ SL := resolveTMEq1(ct,cm)
+ not EQ(SL,'failed)
+ ct=cm
+ not b =>
+ TL := [ct,argt,:TL]
+ t := argt and last argt
+ b and
+ t := resolveTMEq2(cm,argm,[ct,argt,:TL])
+ if t then for p in SL repeat $Subst := augmentSub(CAR p,CDR p,$Subst)
+ t
+
+resolveTMSpecial(t,m) ==
+ -- a few special cases
+ t = $AnonymousFunction and m is ['Mapping,:.] => m
+ t is ['Variable,x] and m is ['OrderedVariableList,le] =>
+ isPatternVar le => ['OrderedVariableList,[x]]
+ PAIRP(le) and member(x,le) => le
+ NIL
+ t is ['Fraction, ['Complex, t1]] and m is ['Complex, m1] =>
+ resolveTM1(['Complex, ['Fraction, t1]], m)
+ t is ['Fraction, ['Polynomial, ['Complex, t1]]] and m is ['Complex, m1] =>
+ resolveTM1(['Complex, ['Fraction, ['Polynomial, t1]]], m)
+ t is ['Mapping,:lt] and m is ['Mapping,:lm] =>
+ #lt ^= #lm => NIL
+ l := NIL
+ ok := true
+ for at in lt for am in lm while ok repeat
+ (ok := resolveTM1(at,am)) => l := [ok,:l]
+ ok and ['Mapping,:reverse l]
+ t is ['Segment,u] and m is ['UniversalSegment,.] =>
+ resolveTM1(['UniversalSegment, u], m)
+ NIL
+
+resolveTMEq1(ct,cm) ==
+ -- ct and cm are type constructors
+ -- tests for a match from cm to ct
+ -- the result is a substitution or 'failed
+ not (CAR ct=CAR cm) => 'failed
+ SL := NIL
+ ct := CDR ct
+ cm := CDR cm
+ b := 'T
+ while ct and cm and b repeat
+ xt := CAR ct
+ ct := CDR ct
+ xm := CAR cm
+ cm := CDR cm
+ if not (atom xm) and CAR xm = ":" -- i.e. Record
+ and CAR xt = ":" and CADR xm = CADR xt then
+ xm := CADDR xm
+ xt := CADDR xt
+ b :=
+ xt=xm => 'T
+ isPatternVar(xm) and
+ p := ASSQ(xm,$Subst) => xt=CDR p
+ p := ASSQ(xm,SL) => xt=CDR p
+ SL := augmentSub(xm,xt,SL)
+ b => SL
+ 'failed
+
+resolveTMEq2(cm,argm,TL) ==
+ -- [cm,argm] is a deconstructed mode,
+ -- TL is a deconstructed type t
+ [ct,argt,:TL] :=
+ $Coerce => bubbleType TL
+ TL
+ null TL and
+ null argm => constructM(ct,argt)
+-- null argm => NIL
+ arg := NIL
+ while argt and argm until not tt repeat
+ x1 := CAR argt
+ argt := CDR argt
+ x2 := CAR argm
+ argm := CDR argm
+ tt := resolveTM1(x1,x2) =>
+ arg := CONS(tt,arg)
+ null argt and null argm and tt and constructM(ct,nreverse arg)
+
+resolveTMRed(t,m) ==
+ -- looks for an applicable rewrite rule at any level of t and tries
+ -- to bubble this constructor up to the top to t
+ TL := NIL
+ until b or not t repeat
+ [ct,:argt] := deconstructT t
+ b := not EQ(t,term1RW(['Resolve,t,m],$ResMode)) and
+ [c0,arg0,:TL0] := bubbleType [ct,argt,:TL]
+ null TL0 and
+ l := term1RWall(['Resolve,constructM(c0,arg0),m],$ResMode)
+ for t0 in l until t repeat t := resolveTMRed1 t0
+ l and t
+ b or
+ TL := [ct,argt,:TL]
+ t := argt and last argt
+ b and t
+
+resolveTMRed1(t) ==
+ -- recursive resolveTMRed which handles all subterms of the form
+ -- (Resolve a b)
+ atom t => t
+ t is ['Resolve,a,b] =>
+ ( a := resolveTMRed1 a ) and ( b := resolveTMRed1 b ) and
+ resolveTM1(a,b)
+ t is ['Incl,a,b] => PAIRP b and member(a,b) and b
+ t is ['Diff,a,b] => PAIRP a and member(b,a) and SETDIFFERENCE(a,[b])
+ t is ['SetIncl,a,b] => PAIRP b and "and"/[member(x,b) for x in a] and b
+ t is ['SetDiff,a,b] => PAIRP b and PAIRP b and
+ intersection(a,b) and SETDIFFERENCE(a,b)
+ t is ['VarEqual,a,b] => (a = b) and b
+ t is ['SetComp,a,b] => PAIRP a and PAIRP b and
+ "and"/[member(x,a) for x in b] and SETDIFFERENCE(a,b)
+ t is ['SimpleAlgebraicExtension,a,b,p] => -- this is a hack. RSS
+ ['SimpleAlgebraicExtension, resolveTMRed1 a, resolveTMRed1 b,p]
+ [( atom x and x ) or resolveTMRed1 x or return NIL for x in t]
+
+--% Type and Mode Representation
+
+eqType(t) ==
+ -- looks for an equivalent but more simple type
+ -- eg, eqType QF I = RN
+ -- the new algebra orginization no longer uses these sorts of types
+-- termRW(t,$TypeEQ)
+ t
+
+equiType(t) ==
+ -- looks for an equivalent but expanded type
+ -- eg, equiType RN == QF I
+ -- the new algebra orginization no longer uses these sorts of types
+-- termRW(t,$TypeEqui)
+ t
+
+getUnderModeOf d ==
+ not PAIRP d => NIL
+-- n := LASSOC(first d,$underDomainAlist) => d.n ----> $underDomainAlist NOW always NIL
+ for a in rest d for m in rest destructT d repeat
+ if m then return a
+
+--deconstructM(t) ==
+-- -- M is a type, which may contain type variables
+-- -- results in a pair (type constructor . mode arguments)
+-- CDR t and constructor? CAR t =>
+-- dt := destructT CAR t
+-- args := [ x for d in dt for y in t | ( x := d and y ) ]
+-- c := [ x for d in dt for y in t | ( x := not d and y ) ]
+-- CONS(c,args)
+-- CONS(t,NIL)
+
+deconstructT(t) ==
+ -- M is a type, which may contain type variables
+ -- results in a pair (type constructor . mode arguments)
+ KDR t and constructor? CAR t =>
+ dt := destructT CAR t
+ args := [ x for d in dt for y in t | ( x := d and y ) ]
+ c := [ x for d in dt for y in t | ( x := not d and y ) ]
+ CONS(c,args)
+ CONS(t,NIL)
+
+constructT(c,A) ==
+ -- c is a type constructor, A a list of argument types
+ A => [if d then POP A else POP c for d in destructT CAR c]
+ c
+
+constructM(c,A) ==
+ -- replaces top level RE's or QF's by equivalent types, if possible
+ containsVars(c) or containsVars(A) => NIL
+ -- collapses illegal FE's
+ CAR(c) = $FunctionalExpression => eqType defaultTargetFE CAR A
+ eqType constructT(c,A)
+
+replaceLast(A,t) ==
+ -- replaces the last element of the nonempty list A by t (constructively
+ nreverse RPLACA(reverse A,t)
+
+destructT(functor)==
+ -- provides a list of booleans, which indicate whether the arguments
+ -- to the functor are category forms or not
+ GETDATABASE(opOf functor,'COSIG)
+
+constructTowerT(t,TL) ==
+ -- t is a type, TL a list of constructors and argument lists
+ -- t is embedded into TL
+ while TL and t repeat
+ [c,arg,:TL] := TL
+ t0 := constructM(c,replaceLast(arg,t))
+ t := canCoerceFrom(t,t0) and t0
+ t
+
+bubbleType(TL) ==
+ -- tries to move the last constructor in TL upwards
+ -- uses canCoerceFrom to test whether two constructors can be bubbled
+ [c1,arg1,:T1] := TL
+ null T1 or null arg1 => TL
+ [c2,arg2,:T2] := T1
+ t := last arg1
+ t2 := constructM(c2,replaceLast(arg2,t))
+ arg1 := replaceLast(arg1,t2)
+ newCanCoerceCommute(c2,c1) or canCoerceCommute(c2, c1) =>
+ bubbleType [c1,arg1,:T2]
+ TL
+
+bubbleConstructor(TL) ==
+ -- TL is a nonempty list of type constructors and nonempty argument
+ -- lists representing a deconstructed type
+ -- then the lowest constructor is bubbled to the top
+ [c,arg,:T1] := TL
+ t := last arg
+ until null T1 repeat
+ [c1,arg1,:T1] := T1
+ arg1 := replaceLast(arg1,t)
+ t := constructT(c1,arg1)
+ constructT(c,replaceLast(arg,t))
+
+compareTT(t1,t2) ==
+ -- 'T if type t1 is more nested than t2
+ -- otherwise 'T if t1 is lexicographically greater than t2
+ EQCAR(t1,$QuotientField) or
+ MEMQ(opOf t2,[$QuotientField, 'SimpleAlgebraicExtension]) => NIL
+ CGREATERP(PRIN2CVEC opOf t1,PRIN2CVEC opOf t2)
+