12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394(*
*
* Copyright (c) 2001-2002,
* John Kodumal <jkodumal@eecs.berkeley.edu>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* 2. 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.
*
* 3. The names of the contributors may not 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.
*
*)exceptionBad_findtype'aurefC=Ecrof'a*int|Linkof'aurefand'auref='aurefCrefletrecfindp=match!pwith|Ecr_->p|Linkp'->letp''=findp'inp:=Linkp'';p''leturefx=ref(Ecr(x,0))letequal(p,p')=(findp==findp')letderefp=match!(findp)with|Ecr(x,_)->x|_->raiseBad_findletupdate(p,x)=letp'=findpinmatch!p'with|Ecr(_,rank)->p':=Ecr(x,rank)|_->raiseBad_findletunifyf(p,q)=letp',q'=findp,findqinmatch(!p',!q')with|(Ecr(px,pr),Ecr(qx,qr))->letx()=f(px,qx)inif(p'==q')thenp':=Ecr(x(),pr)elseifpr==qrthen(p':=Linkq';q':=Ecr(x(),qr+1))elseifpr<qrthen(p':=Linkq';q':=Ecr(x(),qr))else(* pr > qr *)(q':=Linkp';p':=Ecr(x(),pr))|_->raiseBad_findletunion(p,q)=letp',q'=findp,findqinmatch(!p',!q')with|(Ecr(px,pr),Ecr(qx,qr))->if(p'==q')then()elseifpr==qrthen(q':=Ecr(qx,qr+1);p':=Linkq')elseifpr<qrthenp':=Linkq'else(* pr > qr *)q':=Linkp'|_->raiseBad_find