| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771 | {    This file is part of the Free Pascal run time library.    Copyright (c) 2005-2006 by the Free Pascal development team    and Gehard Scholz    It contains the Free Pascal BCD implementation    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS for A PARTICULAR PURPOSE. **********************************************************************}{ "Programming is the time between two bugs" }{     (last words of the unknown programmer) }{ this program was a good test for the compiler: some bugs have been found.  1. WITH in inline funcs produces a compiler error AFTER producing an .exe file     (was already known; I didn't see it in the bug list)  2. macro names were checked for being a keyword, even when starting with     an '_' (produces range check when compiler is compiled with { $r+ }-mode     fixed.  3. { $define program } was not possible in { $macro on } mode     (keywords not allowed: doesn't make sense here)     fixed.  4. the Inc/Dec ( unsigned, signed ) problem (has been similar in the     bug list already)  5. when the result of an overloaded (inline) operator is ABSOLUTEd:     compiler error 200110205     happens only when operator is defined in a unit.  6. two range check errors in scanner.pas     a) array subscripting     b) value out ouf range}{ $define debug_version}{$r+,q+,s+}{ $r-,q-,s-}{$mode objfpc}{$h-}{$macro on}{$define BCDMaxDigits := 64 } { should be even }{ the next defines must be defined by hand,  unless someone shows me a way how to to it with macros }{$define BCDgr4}   { define this if MCDMaxDigits is greater 4,   else undefine! }{$define BCDgr9}   { define this if MCDMaxDigits is greater 9,   else undefine! }{$define BCDgr18}  { define this if MCDMaxDigits is greater 18,  else undefine! }{ $define BCDgr64}  { define this if MCDMaxDigits is greater 64,  else undefine! }{ $define BCDgr180} { define this if MCDMaxDigits is greater 180, else undefine! }{$ifdef BCDgr4} {$note BCD Digits > 4}{$endif}{$ifdef BCDgr9} {$note BCD Digits > 9}{$endif}{$ifdef BCDgr18} {$note BCD Digits > 18}{$endif}{$ifdef BCDgr64} {$note BCD Digits > 64}{$endif}{$ifdef BCDgr180} {$note BCD Digits > 180}{$endif}{ $smartlink on}{ $define some_inlines} { disable this to disallow INLNE }{$define some_packed} { enable this to keep some local structures PACKED }{ $define as_object} { to define the tBCD record as object instead;                      fields then are private  }                     { not done yet! }{$define additional_routines} { to create additional routines and operators }(* only define one of them! *){ $define integ32}{$define integ64}(* only define one of them! *){ $define real8}{$define real10}{check}{$ifndef integ32}  {$ifndef integ64}    {$define integ64}  {$endif}{$endif}{$ifdef integ32}  {$ifdef integ64}    {$undef integ32}  {$endif}{$endif}{check}{$ifndef real8}  {$ifndef real10}    {$define real8}  {$endif}{$endif}{$ifdef real8}  {$ifdef real10}    {$undef real10}  {$endif}{$endif}{$ifdef some_inlines}  {$define make_inline := inline;}{$else}  {$define make_inline := (**)}{$endif}{$ifdef some_packed}  {$define maybe_packed := packed}{$else}  {$define maybe_packed := (**)}{$endif}UNIT FmtBCD;INTERFACE  USES    SysUtils,{    dateutils,}    Variants;  const    MaxStringDigits = 100;          { not used ! }    _NoDecimal = -255;              { not used ! }    _DefaultDecimals = 10;          { not used ! }  { From DB.pas }  { Max supported by Midas }               { must be EVEN }    MaxFmtBCDFractionSize = BCDMaxDigits + Ord ( Odd ( BCDMaxDigits ) );  { Max supported by Midas }    MaxFmtBCDDigits =   32;         { not used ! }    DefaultFmtBCDScale = 6;         { not used ! }    MaxBCDPrecision =   18;         { not used ! }    MaxBCDScale     =   4;          { not used ! }{$ifdef BCDgr64}{ $fatal big 1}  {$define bigger_BCD}  { must be defined                          if MaxFmtBCDFractionSize > 64 }                        { not usable in the moment }{$endif}{$ifdef BCDgr180}{ $fatal big 2}  type    FmtBCDStringtype = AnsiString;  {$define use_Ansistring}{$else}  type    FmtBCDStringtype = string [ 255 ];  {$undef use_Ansistring}{$endif}{$ifdef use_ansistring}  {$note ansi}{$else}  {$note -ansi}{$endif}{$ifdef integ32}  {$define myInttype := LongInt}{$endif}{$ifdef integ64}  {$define myInttype := int64}{$endif}{$ifdef real8}  {$define myRealtype := double}{$endif}{$ifdef real10}  {$define myRealtype := extended}{$endif}{$ifdef SUPPORT_COMP}    {$define comproutines}{$endif SUPPORT_COMP}{$define __low_Fraction := 0 }{$define __high_Fraction := ( ( MaxFmtBCDFractionSize DIV 2 ) - 1 ) }  type    pBCD = ^ tBCD;    tBCD = packed {$ifdef as_object} OBJECT {$else} record {$endif}            {$ifdef as_object} PRIVATE {$endif}             Precision : 0..maxfmtbcdfractionsize;  { 1 (joke?)..64 }{$ifndef bigger_BCD}             SignSpecialPlaces : Byte;      { Sign:1, Special:1, Places:6 }{$else}             Negativ : Boolean;{             Special : Boolean;}             Places : 0..maxfmtbcdfractionsize - 1;{$endif}             Fraction : packed array [ __low_Fraction..__high_Fraction ] of Byte;                            { BCD Nibbles, 00..99 per Byte, high Nibble 1st }            end;  type    tDecimalPoint = ( DecimalPoint_is_Point, DecimalPoint_is_Comma, DecimalPoint_is_System );{ Exception classes }  type    eBCDException = CLASS ( Exception );    eBCDOverflowException = CLASS ( eBCDException );    eBCDNotImplementedException = CLASS ( eBCDException );  var    DecimalPoint : tDecimalPoint = DecimalPoint_is_Point;{ Utility functions for TBCD access }  function BCDPrecision ( const BCD : tBCD ) : Word; make_Inline  function BCDScale ( const BCD : tBCD ) : Word; make_Inline  function IsBCDNegative ( const BCD : tBCD ) : Boolean; make_Inline{ BCD Arithmetic}  procedure BCDNegate ( var BCD : tBCD ); make_Inline{ !!!!!!!!!! most routines are intentionally NOT inline !!!!!!!!!! }{ Returns True if successful, False if Int Digits needed to be truncated }  function NormalizeBCD ( const InBCD : tBCD;                            var OutBCD : tBCD;                          const Prec,                                Scale : Word ) : Boolean;  procedure BCDAdd ( const BCDin1,                           BCDin2 : tBCD;                       var BCDout : tBCD );  procedure BCDSubtract ( const BCDin1,                                BCDin2 : tBCD;                            var BCDout : tBCD );  procedure BCDMultiply ( const BCDin1,                                BCDin2 : tBCD;                            var BCDout : tBCD );  procedure BCDMultiply ( const BCDIn : tBCD;                          const DoubleIn : myRealtype;                            var BCDout : tBCD ); make_Inline  procedure BCDMultiply ( const BCDIn : tBCD;                        const StringIn : FmtBCDStringtype;                          var BCDout : tBCD ); make_Inline{ !!! params changed to const, shouldn't give a problem }  procedure BCDMultiply ( const StringIn1,                                StringIn2 : FmtBCDStringtype;                          var BCDout : tBCD ); make_Inline  procedure BCDDivide ( const Dividend,                              Divisor : tBCD;                          var BCDout : tBCD );  procedure BCDDivide ( const Dividend : tBCD;                        const Divisor : myRealtype;                          var BCDout : tBCD ); make_Inline  procedure BCDDivide ( const Dividend : tBCD;                        const Divisor : FmtBCDStringtype;                            var BCDout : tBCD ); make_Inline{ !!! params changed to const, shouldn't give a problem }  procedure BCDDivide ( const Dividend,                              Divisor : FmtBCDStringtype;                          var BCDout : tBCD ); make_Inline{ TBCD variant creation utils }  procedure VarFmtBCDCreate (   var aDest : Variant;                              const aBCD : tBCD );  function VarFmtBCDCreate : Variant;  function VarFmtBCDCreate ( const aValue : FmtBCDStringtype;                                   Precision,                                   Scale : Word ) : Variant;  function VarFmtBCDCreate ( const aValue : myRealtype;                                   Precision : Word = 18;                                   Scale : Word = 4 ) : Variant;  function VarFmtBCDCreate ( const aBCD : tBCD ) : Variant;  function VarIsFmtBCD ( const aValue : Variant ) : Boolean;  function VarFmtBCD : TVartype;{ Convert string/Double/Integer to BCD struct }  function StrToBCD ( const aValue : FmtBCDStringtype ) : tBCD;  function TryStrToBCD ( const aValue : FmtBCDStringtype;                           var BCD : tBCD ) : Boolean;  function DoubleToBCD ( const aValue : myRealtype ) : tBCD; make_Inline  procedure DoubleToBCD ( const aValue : myRealtype;                            var BCD : tBCD );  function IntegerToBCD ( const aValue : myInttype ) : tBCD;  function VarToBCD ( const aValue : Variant ) : tBCD;{ From DB.pas }  function CurrToBCD ( const Curr : currency;                         var BCD : tBCD;                             Precision : Integer = 32;                             Decimals : Integer = 4 ) : Boolean;{ Convert BCD struct to string/Double/Integer }  function BCDToStr ( const BCD : tBCD ) : FmtBCDStringtype;  function BCDToDouble ( const BCD : tBCD ) : myRealtype;  function BCDToInteger ( const BCD : tBCD;                                Truncate : Boolean = False ) : myInttype;{ From DB.pas }  function BCDToCurr ( const BCD : tBCD;                         var Curr : currency ) : Boolean;{ Formatting BCD as string }  function BCDToStrF ( const BCD : tBCD;                             Format : TFloatFormat;                       const Precision,                             Digits : Integer ) : FmtBCDStringtype;  function FormatBCD ( const Format : string;                             BCD : tBCD ) : FmtBCDStringtype;{ returns -1 if BCD1 < BCD2, 0 if BCD1 = BCD2, 1 if BCD1 > BCD2 }  function BCDCompare ( const BCD1,                              BCD2 : tBCD ) : Integer;{$ifdef additional_routines}  function CurrToBCD ( const Curr : currency ) : tBCD; make_Inline{$ifdef comproutines}  function CompToBCD ( const Curr : Comp ) : tBCD; make_Inline  function BCDToComp ( const BCD : tBCD ) : Comp; make_Inline{$endif}  procedure BCDAdd ( const BCDIn : tBCD;                     const IntIn : myInttype;                       var BCDout : tBCD );  procedure BCDAdd ( const IntIn : myInttype;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline  procedure BCDAdd ( const BCDIn : tBCD;                     const DoubleIn : myRealtype;                       var BCDout : tBCD ); make_Inline  procedure BCDAdd ( const DoubleIn : myRealtype;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline  procedure BCDAdd ( const BCDIn : tBCD;                     const Currin : currency;                       var BCDout : tBCD ); make_Inline  procedure BCDAdd ( const Currin : currency;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline{$ifdef comproutines}  procedure BCDAdd ( const BCDIn : tBCD;                     const Compin : Comp;                       var BCDout : tBCD ); make_Inline  procedure BCDAdd ( const Compin : Comp;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline{$endif}  procedure BCDAdd ( const BCDIn : tBCD;                     const StringIn : FmtBCDStringtype;                       var BCDout : tBCD ); make_Inline  procedure BCDAdd ( const StringIn : FmtBCDStringtype;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline  procedure BCDAdd ( const StringIn1,                           StringIn2 : FmtBCDStringtype;                       var BCDout : tBCD ); make_Inline  procedure BCDSubtract ( const BCDIn : tBCD;                          const IntIn : myInttype;                            var BCDout : tBCD );  procedure BCDSubtract ( const IntIn : myInttype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline  procedure BCDSubtract ( const BCDIn : tBCD;                          const DoubleIn : myRealtype;                            var BCDout : tBCD ); make_Inline  procedure BCDSubtract ( const DoubleIn : myRealtype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline  procedure BCDSubtract ( const BCDIn : tBCD;                          const Currin : currency;                            var BCDout : tBCD ); make_Inline  procedure BCDSubtract ( const Currin : currency;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline{$ifdef comproutines}  procedure BCDSubtract ( const BCDIn : tBCD;                          const Compin : Comp;                            var BCDout : tBCD ); make_Inline  procedure BCDSubtract ( const Compin : Comp;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline{$endif}  procedure BCDSubtract ( const BCDIn : tBCD;                          const StringIn : FmtBCDStringtype;                            var BCDout : tBCD ); make_Inline  procedure BCDSubtract ( const StringIn : FmtBCDStringtype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline  procedure BCDSubtract ( const StringIn1,                                StringIn2 : FmtBCDStringtype;                          var BCDout : tBCD ); make_Inline  procedure BCDMultiply ( const BCDIn : tBCD;                          const IntIn : myInttype;                            var BCDout : tBCD );  procedure BCDMultiply ( const IntIn : myInttype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline  procedure BCDMultiply ( const DoubleIn : myRealtype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline  procedure BCDMultiply ( const BCDIn : tBCD;                          const Currin : currency;                            var BCDout : tBCD ); make_Inline  procedure BCDMultiply ( const Currin : currency;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline{$ifdef comproutines}  procedure BCDMultiply ( const BCDIn : tBCD;                          const Compin : Comp;                            var BCDout : tBCD ); make_Inline  procedure BCDMultiply ( const Compin : Comp;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline{$endif}  procedure BCDMultiply ( const StringIn : FmtBCDStringtype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline  procedure BCDDivide ( const Dividend : tBCD;                        const Divisor : myInttype;                          var BCDout : tBCD ); make_Inline  procedure BCDDivide ( const Dividend : myInttype;                        const Divisor : tBCD;                          var BCDout : tBCD ); make_Inline  procedure BCDDivide ( const Dividend : myRealtype;                        const Divisor : tBCD;                            var BCDout : tBCD ); make_Inline  procedure BCDDivide ( const BCDIn : tBCD;                        const Currin : currency;                          var BCDout : tBCD ); make_Inline  procedure BCDDivide ( const Currin : currency;                        const BCDIn : tBCD;                          var BCDout : tBCD ); make_Inline{$ifdef comproutines}  procedure BCDDivide ( const BCDIn : tBCD;                        const Compin : Comp;                          var BCDout : tBCD ); make_Inline  procedure BCDDivide ( const Compin : Comp;                        const BCDIn : tBCD;                          var BCDout : tBCD ); make_Inline{$endif}  procedure BCDDivide ( const Dividend : FmtBCDStringtype;                        const Divisor : tBCD;                            var BCDout : tBCD ); make_Inline  operator = ( const BCD1,                     BCD2 : tBCD ) z : Boolean; make_Inline  operator < ( const BCD1,                     BCD2 : tBCD ) z : Boolean; make_Inline  operator > ( const BCD1,                     BCD2 : tBCD ) z : Boolean; make_Inline  operator <= ( const BCD1,                      BCD2 : tBCD ) z : Boolean; make_Inline  operator >= ( const BCD1,                      BCD2 : tBCD ) z : Boolean; make_Inline(* ########################            not allowed: why?  operator + ( const BCD : tBCD ) z : tBCD; make_Inline##################################################### *)  operator - ( const BCD : tBCD ) z : tBCD; make_Inline  operator + ( const BCD1,                     BCD2 : tBCD ) z : tBCD; make_Inline  operator + ( const BCD : tBCD;               const i : myInttype ) z : tBCD; make_Inline  operator + ( const i : myInttype;               const BCD : tBCD ) z : tBCD; make_Inline  operator + ( const BCD : tBCD;               const r : myRealtype ) z : tBCD; make_Inline  operator + ( const r : myRealtype;               const BCD : tBCD ) z : tBCD; make_Inline  operator + ( const BCD : tBCD;               const c : currency ) z : tBCD; make_Inline  operator + ( const c : currency;               const BCD : tBCD ) z : tBCD; make_Inline{$ifdef comproutines}  operator + ( const BCD : tBCD;               const c : Comp ) z : tBCD; make_Inline  operator + ( const c : Comp;               const BCD : tBCD ) z : tBCD; make_Inline{$endif}  operator + ( const BCD : tBCD;               const s : FmtBCDStringtype ) z : tBCD; make_Inline  operator + ( const s : FmtBCDStringtype;               const BCD : tBCD ) z : tBCD; make_Inline  operator - ( const BCD1,                     BCD2 : tBCD ) z : tBCD; make_Inline  operator - ( const BCD : tBCD;               const i : myInttype ) z : tBCD; make_Inline  operator - ( const i : myInttype;               const BCD : tBCD ) z : tBCD; make_Inline  operator - ( const BCD : tBCD;               const r : myRealtype ) z : tBCD; make_Inline  operator - ( const r : myRealtype;               const BCD : tBCD ) z : tBCD; make_Inline  operator - ( const BCD : tBCD;               const c : currency ) z : tBCD; make_Inline  operator - ( const c : currency;               const BCD : tBCD ) z : tBCD; make_Inline{$ifdef comproutines}  operator - ( const BCD : tBCD;               const c : Comp ) z : tBCD; make_Inline  operator - ( const c : Comp;               const BCD : tBCD ) z : tBCD; make_Inline{$endif}  operator - ( const BCD : tBCD;               const s : FmtBCDStringtype ) z : tBCD; make_Inline  operator - ( const s : FmtBCDStringtype;               const BCD : tBCD ) z : tBCD; make_Inline  operator * ( const BCD1,                     BCD2 : tBCD ) z : tBCD; make_Inline  operator * ( const BCD : tBCD;               const i : myInttype ) z : tBCD; make_Inline  operator * ( const i : myInttype;               const BCD : tBCD ) z : tBCD; make_Inline  operator * ( const BCD : tBCD;               const r : myRealtype ) z : tBCD; make_Inline  operator * ( const r : myRealtype;               const BCD : tBCD ) z : tBCD; make_Inline  operator * ( const BCD : tBCD;               const c : currency ) z : tBCD; make_Inline  operator * ( const c : currency;               const BCD : tBCD ) z : tBCD; make_Inline{$ifdef comproutines}  operator * ( const BCD : tBCD;               const c : Comp ) z : tBCD; make_Inline  operator * ( const c : Comp;               const BCD : tBCD ) z : tBCD; make_Inline{$endif}  operator * ( const BCD : tBCD;               const s : FmtBCDStringtype ) z : tBCD; make_Inline  operator * ( const s : FmtBCDStringtype;               const BCD : tBCD ) z : tBCD; make_Inline  operator / ( const BCD1,                     BCD2 : tBCD ) z : tBCD; make_Inline  operator / ( const BCD : tBCD;               const i : myInttype ) z : tBCD; make_Inline  operator / ( const i : myInttype;               const BCD : tBCD ) z : tBCD; make_Inline  operator / ( const BCD : tBCD;               const r : myRealtype ) z : tBCD; make_Inline  operator / ( const r : myRealtype;               const BCD : tBCD ) z : tBCD; make_Inline  operator / ( const BCD : tBCD;               const c : currency ) z : tBCD; make_Inline  operator / ( const c : currency;               const BCD : tBCD ) z : tBCD; make_Inline{$ifdef comproutines}  operator / ( const BCD : tBCD;               const c : Comp ) z : tBCD; make_Inline  operator / ( const c : Comp;               const BCD : tBCD ) z : tBCD; make_Inline{$endif}  operator / ( const BCD : tBCD;               const s : FmtBCDStringtype ) z : tBCD; make_Inline  operator / ( const s : FmtBCDStringtype;               const BCD : tBCD ) z : tBCD; make_Inline  operator := ( const i : Byte ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : Byte; make_Inline  operator := ( const i : Word ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : Word; make_Inline  operator := ( const i : longword ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : longword; make_Inline{$if declared ( qword ) }  operator := ( const i : qword ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : qword; make_Inline{$endif}  operator := ( const i : ShortInt ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : ShortInt; make_Inline  operator := ( const i : smallint ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : smallint; make_Inline  operator := ( const i : LongInt ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : LongInt; make_Inline{$if declared ( int64 ) }  operator := ( const i : int64 ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : int64; make_Inline{$endif}  operator := ( const r : Single ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : Single; make_Inline  operator := ( const r : Double ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : Double; make_Inline{$if sizeof ( extended ) <> sizeof ( double )}  operator := ( const r : Extended ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : Extended; make_Inline{$endif}  operator := ( const c : currency ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : currency; make_Inline{$ifdef comproutines}  operator := ( const c : Comp ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : Comp; make_Inline{$endif}  operator := ( const s : string ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : string; make_Inline  operator := ( const s : AnsiString ) z : tBCD; make_Inline  operator := ( const BCD : tBCD ) z : AnsiString; make_Inline{$endif}  function __get_null : tBCD; make_Inline  function __get_one : tBCD; make_Inline  PROPERTY    NullBCD : tBCD Read __get_null;    OneBCD : tBCD Read __get_one;//{$define __lo_bh := 1 * ( -( MaxFmtBCDFractionSize * 1 + 2 ) ) }//{$define __hi_bh := 1 * ( MaxFmtBCDFractionSize * 1 + 1 ) }{$define helper_declarations :=  const    __lo_bh = -( MaxFmtBCDFractionSize + 2 );    __hi_bh =  ( MaxFmtBCDFractionSize + 1 );  type    tBCD_helper = Maybe_Packed record                    Prec : {$ifopt r+} 0..( __hi_bh - __lo_bh + 1 ) {$else} Integer {$endif};                    Plac : {$ifopt r+} 0..( __hi_bh - __lo_bh + 1 ) {$else} Integer {$endif};                    FDig,                    LDig : {$ifopt r+} __lo_bh..__hi_bh {$else} Integer {$endif};                    Singles : Maybe_packed array [ __lo_bh..__hi_bh ]                                of {$ifopt r+} 0..9 {$else} Byte {$endif};                    Neg : Boolean;                   end;    { in the tBCD_helper the bcd is stored for computations,      shifted to the right position }{ {$define __lo_bhb := 1 * ( __lo_bh + __lo_bh ) } }{ {$define __hi_bhb := 1 * ( __hi_bh + __hi_bh + 1 ) } }  const    __lo_bhb = __lo_bh + __lo_bh - 1;    __hi_bhb = __hi_bh + __hi_bh;  type    tBCD_helper_big = Maybe_Packed record                        Prec : {$ifopt r+} 0.. ( __hi_bhb - __lo_bhb + 1 ) {$else} Integer {$endif};                        Plac : {$ifopt r+} 0.. ( __hi_bhb - __lo_bhb + 1 ) {$else} Integer {$endif};                        FDig,                        LDig : {$ifopt r+} __lo_bhb..__hi_bhb {$else} Integer {$endif};                        Singles : Maybe_packed array [ __lo_bhb..__hi_bhb ]                                    of {$ifopt r+} 0 * 0..9 * 9 * Pred ( MaxFmtBCDDigits ) {$else} Integer {$endif};                        Neg : Boolean;                   end;}{$ifdef debug_version}  helper_declarations  procedure unpack_BCD ( const BCD : tBCD;                           var bh : tBCD_helper );  function pack_BCD ( var bh : tBCD_helper;                      var BCD : tBCD ) : Boolean;  procedure dumpBCD ( const v : tBCD );{$endif}IMPLEMENTATION  USES    classes;  type    TFMTBcdFactory = CLASS(TPublishableVarianttype)    PROTECTED      function GetInstance(const v : TVarData): tObject; OVERRIDE;    PUBLIC      procedure BinaryOp(var Left: TVarData; const Right: TVarData; const Operation: TVarOp);    end;    TFMTBcdVarData = CLASS(TPersistent)    PRIVATE      FBcd : tBCD;    PUBLIC      constructor create;      constructor create(const BCD : tBCD);      PROPERTY BCD : tBCD Read FBcd Write FBcd;    end;  var    NullBCD_ : tBCD;    OneBCD_ : tBCD;  function __get_null : tBCD; make_Inline    begin      __get_null := NullBCD_;     end;  function __get_one : tBCD; make_Inline    begin      __get_one := OneBCD_;     end;  type    range_digits = 1..maxfmtbcdfractionsize;    range_digits0 = 0..maxfmtbcdfractionsize;    range_fracdigits = 0..pred ( MaxFmtBCDFractionSize );{$ifopt r+}  var    rcheck : 0..0;    rbad : Byte = 1;{$endif}{$ifndef debug_version}  helper_declarations{$endif}  var    null_ : record              case Boolean of                False: ( bh : tBCD_helper );                True: ( bhb : tBCD_helper_big );             end;    FMTBcdFactory : TFMTBcdFactory = NIL;{$ifndef bigger_BCD}  const    NegBit = 1 SHL 7;    SpecialBit = 1 SHL 6;    PlacesMask = $ff XOR ( NegBit OR SpecialBit );{$endif}{$define _select := {$define _when := if {$define _when := end else if } }                    {$define _then := then begin }                    {$define _whenother := end else begin }                    {$define _endselect := end }  }{$ifdef debug_version}  procedure dumpBCD ( const v : tBCD );    var      i,      j : Integer;    const      ft : ARRAY [ Boolean ] of Char = ( 'f', 't' );    begin{$ifndef bigger_BCD}      Write ( 'Prec:', v.Precision, ' ',              'Neg:', ft[( v.SignSpecialPlaces AND NegBit ) <> 0], ' ',              'Special:', ft[( v.SignSpecialPlaces AND SpecialBit ) <> 0], ' ',              'Places:', v.SignSpecialPlaces AND PlacesMask, ' ' );{$else}      Write ( 'Prec:', v.Precision, ' ',              'Neg:', ft[v.Negativ], ' ',              'Places:', v.Places, ' ' );{$endif}      j := 0;      for i := 1 TO v.Precision do        if Odd ( i )          then Write ( ( v.Fraction[j] AND $f0 ) SHR 4 )          else begin            Write ( v.Fraction[j] AND $0f );            Inc ( j );           end;      WriteLn;     end;  procedure dumpbh ( const v : tBCD_helper );    var      i : Integer;    const      ft : ARRAY [ Boolean ] of Char = ( 'f', 't' );    begin      Write ( 'Prec:', v.Prec, ' ',              'Neg:', ft[v.Neg], ' ',              'Places:', v.Plac, ' ',              'FDig:', v.FDig, ' ',              'LDig:', v.LDig, ' ',              'Digits:', v.LDig - v.FDig + 1, ' ' );      for i := v.FDig TO v.LDig do        Write ( v.Singles[i] );      WriteLn;     end;{$endif}{$if sizeof ( integer ) = 2 }  {$ifdef BCDgr4 }                                  var                                    myMinIntBCD : tBCD;  {$endif}{$else}  {$if sizeof ( integer ) = 4 }    {$ifdef BCDgr9 }                                  var                                    myMinIntBCD : tBCD;    {$endif}  {$else}    {$if sizeof ( integer ) = 8 }      {$ifdef BCDgr18 }                                  var                                    myMinIntBCD : tBCD;      {$endif}    {$else}      {$fatal You have an interesting integer type! Sorry, not supported}    {$endif}  {$endif}{$endif}  procedure not_implemented;    begin      RAISE eBCDNotImplementedException.create ( 'not implemented' );     end;  procedure unpack_BCD ( const BCD : tBCD;                           var bh : tBCD_helper );    var      i : {$ifopt r+} __lo_bh + 1 ..__hi_bh {$else} Integer {$endif};      j : {$ifopt r+} -1..__high_fraction {$else} Integer {$endif};      vv : {$ifopt r+} $00..$99 {$else} Integer {$endif};    begin      bh := null_.bh;      WITH bh,           BCD do        begin          Prec := Precision;          if Prec > 0            then begin{$ifndef bigger_BCD}              Plac := SignSpecialPlaces AND PlacesMask;              Neg := ( SignSpecialPlaces AND NegBit ) <> 0;{$else}              Plac := Places;              Neg := Negativ;{$endif}              LDig := Plac;              FDig := LDig - Prec + 1;              j := -1;              i := FDig;              while i <= LDig do                begin                  Inc ( j );                  vv := Fraction[j];                  Singles[i] := ( vv {AND $f0} ) SHR 4;                  if i < LDig                    then Singles[i+1] := vv AND $0f;                 Inc ( i, 2 );                 end;             end;         end;     end;  function pack_BCD ( var bh : tBCD_helper;                      var BCD : tBCD ) : Boolean;  { return TRUE if successful (BCD valid) }    var      pre :    {$ifopt r+} 0..__hi_bh - __lo_bh + 1 {$else} Integer {$endif};      fra :    {$ifopt r+} -1 * ( __hi_bh - __lo_bh + 1 )..__hi_bh - __lo_bh + 1 {$else} Integer {$endif};      tm :     {$ifopt r+} 0..__hi_bh - __lo_bh + 1 - Pred ( MaxFmtBCDFractionSize ) {$else} Integer {$endif};      i :      {$ifopt r+} low ( bh.FDig ) - 1..high ( bh.LDig ) {$else} Integer {$endif};      rp :     {$ifopt r+} low ( BCD.Fraction )..high ( BCD.Fraction ) + 1 {$else} Integer {$endif};      ue :     {$ifopt r+} 0..1 {$else} Integer {$endif};      v :      {$ifopt r+} 0..10 {$else} Integer {$endif};      lnz :    {$ifopt r+} low ( bh.FDig )..high ( bh.LDig ) {$else} Integer {$endif};      doround,      lnzf : Boolean;    begin      pack_BCD := False;      BCD := NullBCD;      WITH BCD,           bh do        begin          lnzf := FDig < 0;          while lnzf do            if Singles[FDig] = 0              then begin                Inc ( FDig );                if FDig = 0                  then lnzf := False;               end              else lnzf := False;          pre := LDig - FDig + 1;          fra := Plac;          doround := False;          if fra >= MaxFmtBCDFractionSize            then begin              doround := True;              tm := fra - Pred ( MaxFmtBCDFractionSize );{             dec ( pre, tm );   Dec/Inc error? }              pre := pre - tm;{             Dec ( fra, tm );   Dec/Inc error? }              fra := fra - tm;{             Dec ( LDig, tm );   Dec/Inc error? }              LDig := LDig - tm;             end;          if pre > MaxFmtBCDFractionSize            then begin              doround := True;              tm := pre - MaxFmtBCDFractionSize;{             Dec ( pre, tm );   Dec/Inc error? }              pre := pre - tm;{             Dec ( fra, tm );   Dec/Inc error? }              fra := fra - tm;{             Dec ( LDig, tm );   Dec/Inc error? }              LDig := LDig - tm;             end;          if fra < 0            then EXIT;          if doround            then begin              v := Singles[fra + 1];              if v > 4                then begin                  ue := 1;                  i := LDig;                  while ( i >= FDig ) AND ( ue <> 0 ) do                    begin                      v := Singles[i] + ue;                      ue := v DIV 10;                      Singles[i] := v MOD 10;                      Dec ( i );                     end;                  if ue <> 0                    then begin                      Dec ( FDig );                      Singles[FDig] := ue;                      Dec ( LDig );                      Dec ( fra );                      if fra < 0                        then EXIT;                     end;                 end;             end;          lnzf := False;          i := LDig;          while ( i >= FDig ) AND ( NOT lnzf ) do            begin              if Singles[i] <> 0                then begin                  lnz := i;                  lnzf := True;                 end;              Dec ( i );             end;          if lnzf            then begin              tm := LDig - lnz;              if tm <> 0                then begin{                 Dec ( pre, tm );   Dec/Inc error? }                  pre := pre - tm;{                 Dec ( fra, tm );   Dec/Inc error? }                  fra := fra - tm;{                 Dec ( LDig, tm );   Dec/Inc error? }                  LDig := LDig - tm;                  if fra < 0                    then begin{                     Dec ( pre, fra );    Dec/Inc error? }                      pre := pre - fra;{                     Dec ( LDig, fra );   Dec/Inc error? }                      LDig := LDig - fra;                      fra := 0;                     end;                 end;             end            else begin              LDig := FDig;              fra := 0;              pre := 0;              Neg := False;             end;          if pre <> 0            then begin              Precision := pre;              rp := 0;              i := FDig;              while i <= LDig do                begin                  if i < LDig                    then Fraction[rp] := ( Singles[i] SHL 4 ) OR Singles[i + 1]                    else Fraction[rp] := Singles[i] SHL 4;                  Inc ( rp );                  Inc ( i, 2 );                 end;{$ifndef bigger_BCD}              if Neg                then SignSpecialPlaces := NegBit;              SignSpecialPlaces := SignSpecialPlaces OR fra;{$else}              Negativ := Neg;              Places := fra;{$endif}             end;         end;      pack_BCD := True;     end;  procedure SetDecimals ( var dp,                              dc : Char );    begin      case DecimalPoint of        DecimalPoint_is_Point: begin                                 dp := '.';                                 dc := ',';                                end;        DecimalPoint_is_Comma: begin                                 dp := ',';                                 dc := '.';                                end;{ find out language-specific ? }        DecimalPoint_is_System: begin                                 dp := DecimalSeparator;                                 dc := ThousandSeparator;                                end;       end;     end;  function BCDPrecision ( const BCD : tBCD ) : Word; make_Inline    begin      BCDPrecision := BCD.Precision;     end;  function BCDScale ( const BCD : tBCD ) : Word; make_Inline    begin{$ifndef bigger_BCD}      BCDScale := BCD.SignSpecialPlaces AND PlacesMask;{$else}      BCDScale := BCD.Places;{$endif}     end;  function IsBCDNegative ( const BCD : tBCD ) : Boolean; make_Inline    begin{$ifndef bigger_BCD}      IsBCDNegative := ( BCD.SignSpecialPlaces AND NegBit ) <> 0;{$else}      IsBCDNegative := BCD.Negativ;{$endif}     end;{ BCD Arithmetic}  procedure BCDNegate ( var BCD : tBCD ); make_Inline    begin{ with-statement geht nicht !!      with bcd do        if precision <> 0          then signspecialplaces := signspecialplaces xor negbit;}        if BCD.Precision <> 0          then{$ifndef bigger_BCD}            BCD.SignSpecialPlaces := BCD.SignSpecialPlaces XOR NegBit;{$else}            BCD.Negativ := NOT BCD.Negativ;{$endif}     end;{ returns -1 if BCD1 < BCD2, 0 if BCD1 = BCD2, 1 if BCD1 > BCD2 }  function BCDCompare ( const BCD1,                              BCD2 : tBCD ) : Integer;    var      pl1 :   {$ifopt r+} 0..maxfmtbcdfractionsize - 1 {$else} Integer {$endif};      pl2 :   {$ifopt r+} 0..maxfmtbcdfractionsize - 1 {$else} Integer {$endif};      pr1 :   {$ifopt r+} 0..maxfmtbcdfractionsize {$else} Integer {$endif};      pr2 :   {$ifopt r+} 0..maxfmtbcdfractionsize {$else} Integer {$endif};      pr :    {$ifopt r+} 0..maxfmtbcdfractionsize {$else} Integer {$endif};      idig1 : {$ifopt r+} 0..maxfmtbcdfractionsize {$else} Integer {$endif};      idig2 : {$ifopt r+} 0..maxfmtbcdfractionsize {$else} Integer {$endif};      i :     {$ifopt r+} __low_Fraction..__high_Fraction + 1 {$else} Integer {$endif};      f1 :    {$ifopt r+} $00..$99 {$else} Integer {$endif};      f2 :    {$ifopt r+} $00..$99 {$else} Integer {$endif};      res :   {$ifopt r+} -1..1 {$else} Integer {$endif};      neg1,      neg2 : Boolean;    begin{$ifndef bigger_BCD}      neg1 := ( BCD1.SignSpecialPlaces AND NegBit ) <> 0;      neg2 := ( BCD2.SignSpecialPlaces AND NegBit ) <> 0;{$else}      neg1 := BCD1.Negativ;      neg2 := BCD2.Negativ;{$endif}      _SELECT        _WHEN neg1 AND ( NOT neg2 )          _THEN result := -1;        _WHEN ( NOT neg1 ) AND neg2          _THEN result := +1;        _WHENOTHER          pr1 := BCD1.Precision;          pr2 := BCD2.Precision;{$ifndef bigger_BCD}          pl1 := BCD1.SignSpecialPlaces AND PlacesMask;          pl2 := BCD2.SignSpecialPlaces AND PlacesMask;{$else}          pl1 := BCD1.Places;          pl2 := BCD2.Places;{$endif}          idig1 := pr1 - pl1;          idig2 := pr2 - pl2;          if idig1 <> idig2            then begin              if ( idig1 > idig2 ) = neg1                then result := -1                else result := +1;             end            else begin              if pr1 < pr2                then pr := pr1                else pr := pr2;              res := 0;              i := __low_Fraction;              while ( res = 0 ) AND ( i < ( __low_Fraction + ( pr DIV 2 ) ) ) do                begin{                  if BCD1.Fraction[i] < BCD2.Fraction[i]                    then res := -1                    else                      if BCD1.Fraction[i] > BCD2.Fraction[i]                        then res := +1;}                  _SELECT                    _WHEN BCD1.Fraction[i] < BCD2.Fraction[i]                      _THEN res := -1                    _WHEN BCD1.Fraction[i] > BCD2.Fraction[i]                      _THEN res := +1;                    _WHENOTHER                   _endSELECT;                  Inc ( i );                 end;              if res = 0                then begin                  if Odd ( pr )                    then begin                      f1 := BCD1.Fraction[i] AND $f0;                      f2 := BCD2.Fraction[i] AND $f0;{                      if f1 < f2                        then res := -1                        else                          if f1 > f2                            then res := +1;}                      _SELECT                        _WHEN f1 < f2                          _THEN res := -1                        _WHEN f1 > f2                          _THEN res := +1;                      _endSELECT;                     end;                 end;              if neg1                then result := 0 - res                else result := res;             end;       _endSELECT     end;{ Convert string/Double/Integer to BCD struct }  function TryStrToBCD ( const aValue : FmtBCDStringtype;                           var BCD : tBCD ) : Boolean;{ shall this return TRUE when error and FALSE when o.k. or the other way round ? }    var{$ifndef use_ansistring}      lav : {$ifopt r+} 0..high ( aValue ) {$else} Integer {$endif};      i   : {$ifopt r+} 0..high ( aValue ) {$else} Integer {$endif};{$else}      lav : {$ifopt r+} longword {$else} longword {$endif};      i   : {$ifopt r+} longword {$else} longword {$endif};{$endif}      ch : Char;      dp,      dc : Char;    type      ife = ( inint, infrac, inexp );{$define max_exp_scanned := 9999 }    var      inife : ife;      lvars : record                fp,                lp : ARRAY [ ife ]{$ifndef use_ansistring}                       of {$ifopt r+} 0..high ( aValue ) {$else} Integer {$endif};                pfnb : {$ifopt r+} 0..high ( aValue ) {$else} Integer {$endif};                ps :   {$ifopt r+} 0..high ( aValue ) {$else} Integer {$endif};                pse :  {$ifopt r+} 0..high ( aValue ) {$else} Integer {$endif};                errp : {$ifopt r+} 0..high ( aValue ) {$else} Integer {$endif};{$else}                       of {$ifopt r+} longword {$else} longword {$endif};                pfnb : {$ifopt r+} longword {$else} longword {$endif};                ps :   {$ifopt r+} longword {$else} longword {$endif};                pse :  {$ifopt r+} longword {$else} longword {$endif};                errp : {$ifopt r+} longword {$else} longword {$endif};{$endif}                exp :  {$ifopt r+} -max_exp_scanned..max_exp_scanned {$else} Integer {$endif};                p :    {$ifopt r+} -max_exp_scanned..max_exp_scanned {$else} Integer {$endif};                bh : tBCD_helper;                nbf : Boolean;               end;    begin      result := False;      FillChar ( lvars, SizeOf ( lvars ), #0 );      BCD := NullBCD;      lav := Length ( aValue );      if lav <> 0        then          WITH lvars,               bh do            begin              SetDecimals ( dp, dc );              while ( pfnb < lav ) AND ( NOT nbf ) do                begin                  Inc ( pfnb );                  nbf := aValue[pfnb] <> ' ';                 end;              if nbf                then begin                  if aValue[pfnb] IN [ '+', '-' ]                    then begin                      ps := pfnb;                      Inc ( pfnb );                     end;                  inife := low ( inife );                  for i := pfnb TO lav do                    begin                      ch := aValue[i];                      case ch of                        '0'..'9': begin                                    case inife of                                      inint,                                      inexp: if fp[inife] = 0                                               then begin                                                 if ch <> '0'                                                   then begin                                                     fp[inife] := i;                                                     lp[inife] := i;                                                    end;                                                end                                               else lp[inife] := i;                                      infrac: begin                                                if fp[infrac] = 0                                                  then fp[infrac] := i;                                                if ch <> '0'                                                  then lp[infrac] := i;                                               end;                                     end;                                   end;                        ',',                        '.': if ch = dp                               then begin                                 if inife <> inint                                   then result := True                                   else inife := infrac;                                end;                        'e',                        'E': if inife = inexp                               then result := True                               else inife := inexp;                        '+',                        '-': if ( inife = inexp ) AND ( fp[inexp] = 0 )                               then pse := i                               else result := True;                        else begin                          result := True;                          errp := i;                         end;                       end;                     end;                  if result                    then begin                      result := False;                      for i := errp TO lav do                        if aValue[i] <> ' '                          then result := True;                     end;                  if result                    then EXIT;                  if ps <> 0                    then Neg := aValue[ps] = '-';                  if lp[infrac] = 0                    then fp[infrac] := 0;                  if fp[inexp] <> 0                    then begin                      exp := 0;                      for i := fp[inexp] TO lp[inexp] do                        if NOT result                          then                            if aValue[i] <> dc                              then begin                                exp := exp * 10 + ( Ord ( aValue[i] ) - Ord ( '0' ) );                                if exp > 999                                  then result := True;                               end;                      if result                        then EXIT;                      if pse <> 0                        then                          if aValue[pse] = '-'                            then exp := -exp;                     end;                  p := -exp;                  if fp[infrac] <> 0                    then begin                      for i := fp[infrac] TO lp[infrac] do                        if aValue[i] <> dc                          then begin                            if p < ( MaxFmtBCDFractionSize + 2 )                              then begin                                Inc ( p );                                Singles[p] := Ord ( aValue[i] ) - Ord ( '0' );                               end;                           end;                     end;                  LDig := p;                  p := 1 - exp;                  if fp[inint] <> 0                    then                      for i := lp[inint] DOWNTO fp[inint] do                        if aValue[i] <> dc                          then begin                            if p > - ( MaxFmtBCDFractionSize + 2 )                              then begin                                Dec ( p );                                Singles[p] := Ord ( aValue[i] ) - Ord ( '0' );                               end                              else result := True;                           end;                  if result                    then EXIT;                  FDig := p;                  if LDig < 0                    then LDig := 0;                  Plac := LDig;                  result := NOT pack_BCD ( bh, BCD );                 end;             end;     end;  function StrToBCD ( const aValue : FmtBCDStringtype ) : tBCD;    var      BCD : tBCD;    begin      if TryStrToBCD ( aValue, BCD )        then begin          RAISE eBCDOverflowException.create ( 'in StrToBCD' );         end        else StrToBCD := BCD;     end;  procedure DoubleToBCD ( const aValue : myRealtype;                            var BCD : tBCD );    var      s : string [ 30 ];      dp : tDecimalPoint;    begin      Str ( aValue : 25, s );      dp := DecimalPoint;      DecimalPoint := DecimalPoint_is_Point;      BCD := StrToBCD ( s );      DecimalPoint := dp;     end;  function DoubleToBCD ( const aValue : myRealtype ) : tBCD; make_Inline    begin      DoubleToBCD ( aValue, result );     end;  function IntegerToBCD ( const aValue : myInttype ) : tBCD;    var      bh : tBCD_helper;      v : {$ifopt r+} 0..high ( myInttype ) {$else} Integer {$endif};      p : {$ifopt r+} low ( bh.Singles ) - 1..0 {$else} Integer {$endif};      Error,      exitloop : Boolean;    begin      _SELECT        _WHEN aValue = 0          _THEN result := NullBCD;        _WHEN aValue = 1          _THEN result := OneBCD;        _WHEN aValue = low ( myInttype )          _THEN{$if declared ( myMinIntBCD ) }            result := myMinIntBCD;{$else}            RAISE eBCDOverflowException.create ( 'in IntegerToBCD' );{$endif}        _WHENOTHER          bh := null_.bh;          WITH bh do            begin              Neg := aValue < 0;              if Neg                then v := -aValue                else v := +aValue;              LDig := 0;              p := 0;              Error := False;              REPEAT                Singles[p] := v MOD 10;                v := v DIV 10;                exitloop := v = 0;                Dec ( p );                if p < low ( Singles )                  then begin                    exitloop := True;                    Error := True;(* what to do if error occured? *)                    RAISE eBCDOverflowException.create ( 'in IntegerToBCD' );                   end;              UNTIL exitloop;              FDig := p + 1;             end;          pack_BCD ( bh, result );       _endSELECT;     end;  function VarToBCD ( const aValue : Variant ) : tBCD;    begin      not_implemented;     end;  function CurrToBCD ( const Curr : currency;                         var BCD : tBCD;                             Precision : Integer = 32;                             Decimals : Integer = 4 ) : Boolean;{  this works under the assumption that a currency is an int64,  except for scale of 10000}    var      i : int64 absolute Curr;    begin      BCD := IntegerToBCD ( i );{$ifndef bigger_BCD}      BCD.SignSpecialPlaces := 4 OR ( BCD.SignSpecialPlaces AND NegBit );{$else}      BCD.Places := 4;{$endif}      CurrToBCD := False;      if Decimals <> 4        then NormalizeBCD ( BCD, BCD, Precision, Decimals );     end;{$ifdef comproutines}  function CompToBCD ( const Curr : Comp ) : tBCD; make_Inline    var      cc : int64 absolute Curr;    begin      result := IntegerToBCD ( cc );     end;  function BCDToComp ( const BCD : tBCD ) : Comp; make_Inline    var      zz : record             case Boolean of               False: ( i : int64 );               True: ( c : Comp );            end;    begin      zz.i := BCDToInteger ( BCD );      BCDToComp := zz.c;     end;{$endif}{ Convert BCD struct to string/Double/Integer }  function BCDToStr ( const BCD : tBCD ) : FmtBCDStringtype;    var      bh : tBCD_helper;      l :  {$ifopt r+} 0..maxfmtbcdfractionsize + 1 + 1 {$else} Integer {$endif};      i :  {$ifopt r+} low ( bh.FDig )..high ( bh.LDig ) {$else} Integer {$endif};      pp : {$ifopt r+} low ( bh.FDig ) - 1..1 {$else} Integer {$endif};    begin{$ifdef use_ansistring}      result := '';{$endif}      unpack_BCD ( BCD, bh );      WITH bh do        begin          l := 0;          if Neg            then begin{$ifndef use_ansistring}              Inc ( l );              result[1] := '-';{$else}              result := result + '-';{$endif}             end;          if Prec = Plac            then begin{$ifndef use_ansistring}              Inc ( l );              result[1] := '0';{$else}              result := result + '0';{$endif}             end;          if Prec > 0            then begin              pp := low ( bh.FDig ) - 1;              if Plac > 0                then pp := 1;              for i := FDig TO LDig do                begin                  if i = pp                    then begin{$ifndef use_ansistring}                      Inc ( l );                      result[l] := '.';{$else}                      result := result + '.';{$endif}                     end;{$ifndef use_ansistring}                  Inc ( l );                  result[l] := Chr ( Singles[i] + Ord ( '0' ) );{$else}                  result := result + Chr ( Singles[i] + Ord ( '0' ) );{$endif}                 end;             end;         end;{$ifndef use_ansistring}      result[0] := Chr ( l );{$endif}     end;  function BCDToDouble ( const BCD : tBCD ) : myRealtype;    var      bh : tBCD_helper;      i : {$ifopt r+} low ( bh.FDig )..high ( bh.LDig ) {$else} Integer {$endif};      r,      e : myRealtype;    begin      unpack_BCD ( BCD, bh );      WITH bh do        begin          r := 0;          e := 1;          for i := 0 DOWNTO FDig do            begin              r := r + Singles[i] * e;              e := e * 10;             end;          e := 1;          for i := 1 TO LDig do            begin              e := e / 10;              r := r + Singles[i] * e;             end;          if Neg            then BCDToDouble := -r            else BCDToDouble := +r;         end;     end;  function BCDToInteger ( const BCD : tBCD;                                Truncate : Boolean = False ) : myInttype;    var      bh : tBCD_helper;      res : myInttype;      i : {$ifopt r+} low ( bh.FDig )..0 {$else} Integer {$endif};{ unclear: behaviour if overflow: abort? return 0? return something? so: checks are missing yet}    begin      unpack_BCD ( BCD, bh );      res := 0;      WITH bh do        begin          for i := FDig TO 0 do            res := res * 10 - Singles[i];          if NOT Truncate            then              if Plac > 0                then                  if Singles[1] > 4                    then Dec ( res );          if Neg            then BCDToInteger := +res            else BCDToInteger := -res;         end;     end;{ From DB.pas }  function BCDToCurr ( const BCD : tBCD;                         var Curr : currency ) : Boolean;    var      bh : tBCD_helper;      res : int64;      c : currency absolute res;      i : {$ifopt r+} low ( bh.FDig )..4 {$else} Integer {$endif};{ unclear: behaviour if overflow: abort? return 0? return something?}    begin      BCDToCurr := True;      unpack_BCD ( BCD, bh );      res := 0;      WITH bh do        begin          for i := FDig TO 4 do            res := res * 10 + Singles[i];          if Plac > 4            then              if Singles[5] > 4                then Inc ( res );          if Neg            then Curr := -c            else Curr := +c;         end;     end;  procedure BCDAdd ( const BCDin1,                           BCDin2 : tBCD;                       var BCDout : tBCD );    var      bhr,      bh1,      bh2 : tBCD_helper;      ue :    {$ifopt r+} 0..1 {$else} Integer {$endif};      i :     {$ifopt r+} low ( bh1.FDig )..high ( bh1.LDig ) {$else} Integer {$endif};      v :     {$ifopt r+} 0..9 + 9 + 1 {$else} Integer {$endif};      BCD : tBCD;      negate : Boolean;    begin      negate := IsBCDNegative ( BCDin1 );      if negate <> IsBCDNegative ( BCDin2 )        then begin          if negate            then begin              BCD := BCDin1;              BCDNegate ( BCD );              BCDSubtract ( BCDin2, BCD, BCDout );              EXIT;             end;          BCD := BCDin2;          BCDNegate ( BCD );          BCDSubtract ( BCDin1, BCD, BCDout );          EXIT;         end;      bhr := null_.bh;      WITH bhr do        begin          unpack_BCD ( BCDin1, bh1 );          unpack_BCD ( BCDin2, bh2 );          if bh1.FDig < bh2.FDig            then FDig := bh1.FDig            else FDig := bh2.FDig;          if bh1.LDig > bh2.LDig            then LDig := bh1.LDig            else LDig := bh2.LDig;          Plac := LDig;          ue := 0;          for i := LDig DOWNTO FDig do            begin              v := bh1.Singles[i] + bh2.Singles[i] + ue;              ue := v DIV 10;              Singles[i] := v MOD 10;             end;          if ue <> 0            then begin              Dec ( FDig );              Singles[FDig] := ue;             end;          Neg := negate;         end;      if NOT pack_BCD ( bhr, BCDout )        then begin          RAISE eBCDOverflowException.create ( 'in BCDAdd' );         end;     end;  procedure BCDSubtract ( const BCDin1,                                BCDin2 : tBCD;                            var BCDout : tBCD );    var      bhr,      bh1,      bh2 : tBCD_helper;      cmp : {$ifopt r+} -1..1 {$else} Integer {$endif};      ue :  {$ifopt r+} 0..1 {$else} Integer {$endif};      i :   {$ifopt r+} low ( bh1.FDig )..high ( bh1.LDig ) {$else} Integer {$endif};      v :   {$ifopt r+} 0 - 9 - 1..9 - 0 - 0 {$else} Integer {$endif};      negate : Boolean;      BCD : tBCD;    begin      negate := IsBCDNegative ( BCDin1 );      if negate <> IsBCDNegative ( BCDin2 )        then begin          if negate            then begin              BCD := BCDin1;              BCDNegate ( BCD );              BCDAdd ( BCDin2, BCD, BCDout );              BCDNegate ( BCDout );              EXIT;             end;          BCD := BCDin2;          BCDNegate ( BCD );          BCDAdd ( BCDin1, BCD, BCDout );          EXIT;         end;      cmp := BCDCompare ( BCDin1, BCDin2 );      if cmp = 0        then begin          BCDout := NullBCD;          EXIT;         end;      bhr := null_.bh;                    {                      n      n }      WITH bhr do                          {      >       <       >      < }        begin                              {                               }          if ( cmp > 0 ) = negate          {   +123     +12     -12   -123 }            then begin                     {  - +12  - +123  - -123  - -12 }              unpack_BCD ( BCDin1, bh2 ); {              x       x        }              unpack_BCD ( BCDin2, bh1 ); {      s       s       s      s }              negate := NOT negate;       {     nn       n      nn      n }             end            else begin              unpack_BCD ( BCDin1, bh1 );              unpack_BCD ( BCDin2, bh2 );             end;          if bh1.FDig < bh2.FDig            then FDig := bh1.FDig            else FDig := bh2.FDig;          if bh1.LDig > bh2.LDig            then LDig := bh1.LDig            else LDig := bh2.LDig;          Plac := LDig;          ue := 0;          for i := LDig DOWNTO FDig do            begin              v := Integer ( bh1.Singles[i] ) - bh2.Singles[i] - ue;              ue := 0;              if v < 0                then begin                  ue := 1;                  Inc ( v, 10 );                 end;              Singles[i] := v;             end;          Neg := negate;          if NOT pack_BCD ( bhr, BCDout )            then begin{should never occur!}              RAISE eBCDOverflowException.create ( 'in BCDSubtract' );             end;         end;     end;{ Returns True if successful, False if Int Digits needed to be truncated }  function NormalizeBCD ( const InBCD : tBCD;                            var OutBCD : tBCD;                          const Prec,                                Scale : Word ) : Boolean;    var      bh : tBCD_helper;      tm : {$ifopt r+} 1..maxfmtbcdfractionsize - 1 {$else} Integer {$endif};    begin      NormalizeBCD := True;{$ifopt r+}      if ( Prec < 0 ) OR ( Prec > MaxFmtBCDFractionSize ) then rcheck := rbad;      if ( Scale < 0 ) OR ( Prec >= MaxFmtBCDFractionSize ) then rcheck := rbad;{$endif}      if BCDScale ( InBCD ) > Scale        then begin          unpack_BCD ( InBCD, bh );          WITH bh do            begin              tm := Plac - Scale;              Plac := Scale;{             dec ( prec, tm );   Dec/Inc error? }              Prec := Prec - tm;{             dec ( ldig, tm );   Dec/Inc error? }              LDig := LDig - tm;              NormalizeBCD := False;             end;          if NOT pack_BCD ( bh, OutBCD )            then begin              RAISE eBCDOverflowException.create ( 'in BCDAdd' );             end;         end;     end;  procedure BCDMultiply ( const BCDin1,                                BCDin2 : tBCD;                            var BCDout : tBCD );    var      bh1,      bh2,      bhr : tBCD_helper;      bhrr : tBCD_helper_big;      i1 : {$ifopt r+} low ( bh1.FDig )..high ( bh1.LDig ) {$else} Integer {$endif};      i2 : {$ifopt r+} low ( bh2.FDig )..high ( bh2.LDig ) {$else} Integer {$endif};      i3 : {$ifopt r+} low ( bhrr.FDig )..high ( bhrr.LDig ) {$else} Integer {$endif};      v : {$ifopt r+} low ( bhrr.Singles[0] )..high ( bhrr.Singles[0] ) {$else} Integer {$endif};      ue : {$ifopt r+} low ( bhrr.Singles[0] ) DIV 10..high ( bhrr.Singles[0] ) DIV 10 {$else} Integer {$endif};    begin      unpack_BCD ( BCDin1, bh1 );      unpack_BCD ( BCDin2, bh2 );      if ( bh1.Prec = 0 ) OR ( bh2.Prec = 0 )        then begin          BCDout := NullBCD;          EXIT;         end;      bhr := null_.bh;      bhrr := null_.bhb;      WITH bhrr do        begin          Neg := bh1.Neg XOR bh2.Neg;{writeln ( __lo_bhb, ' ', __hi_bhb, ' ', bh1.fdig, ' ', bh2.fdig, ' ', low ( fdig ), ' ', low ( ldig ) );}          FDig := bh1.FDig + bh2.FDig;          LDig := bh1.LDig + bh2.LDig;          for i1 := bh1.FDig TO bh1.LDig do            for i2 := bh2.FDig TO bh2.LDig dobegin              Inc ( Singles[i1 + i2],                    bh1.Singles[i1]                    * bh2.Singles[i2] );{write ( Singles[i1 + i2], ' ', bh1.Singles[i1], ' ', bh2.Singles[i2], ' : ' );writeln ( Singles[i1 + i2] + bh1.Singles[i1] + bh2.Singles[i2] );}{              Singles[i1 + i2] := Singles[i1 + i2]                                       + bh1.Singles[i1]                                         * bh2.Singles[i2];}end;{for i3 := fdig to ldig do  write ( ' ', singles[i3] );writeln;}          if FDig < low ( bhr.Singles )            then RAISE eBCDOverflowException.create ( 'in BCDMultiply' );          ue := 0;          for i3 := LDig DOWNTO FDig do            begin              v := Singles[i3] + ue;              ue := v DIV 10;              v := v MOD 10;              bhr.Singles[i3] := v;             end;          while ue <> 0 do            begin              Dec ( FDig );              if FDig < low ( bhr.Singles )                then RAISE eBCDOverflowException.create ( 'in BCDMultiply' );              bhr.Singles[FDig] := ue MOD 10;              ue := ue DIV 10;             end;          bhr.Plac := LDig;          bhr.FDig := FDig;          if LDig > high ( bhr.Singles )            then bhr.LDig := high ( bhr.Singles )            else bhr.LDig := LDig;         end;      if NOT pack_BCD ( bhr, BCDout )        then begin          RAISE eBCDOverflowException.create ( 'in BCDMultiply' );         end;     end;  procedure BCDMultiply ( const BCDIn : tBCD;                          const DoubleIn : myRealtype;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( BCDIn, DoubleToBCD ( DoubleIn ), BCDout );     end;  procedure BCDMultiply ( const BCDIn : tBCD;                          const StringIn : FmtBCDStringtype;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( BCDIn, StrToBCD ( StringIn ), BCDout );     end;  procedure BCDMultiply ( const StringIn1,                                StringIn2 : FmtBCDStringtype;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( StrToBCD ( StringIn1 ), StrToBCD ( StringIn2 ), BCDout );     end;  procedure BCDDivide ( const Dividend,                              Divisor : tBCD;                          var BCDout : tBCD );    var      bh1 : ARRAY [ Boolean ] of tBCD_helper;      bh2,      bh : tBCD_helper;      p :     {$ifopt r+} low ( bh.FDig ) - high ( bh.FDig )..high ( bh.FDig ) - low ( bh.FDig ) {$else} Integer {$endif};      v1 :    {$ifopt r+} low ( bh.Singles[0] )..high ( bh.Singles[0] ) {$else} Integer {$endif};      v2 :    {$ifopt r+} low ( bh.Singles[0] )..high ( bh.Singles[0] ) {$else} Integer {$endif};      lFDig : {$ifopt r+} low ( bh.FDig )..high ( bh.FDig ) {$else} Integer {$endif};      d1 :    {$ifopt r+} low ( bh.LDig ) - high ( bh.FDig )..high ( bh.LDig ) - low ( bh.FDig ) {$else} Integer {$endif};      d2 :    {$ifopt r+} low ( bh.LDig ) - high ( bh.FDig )..high ( bh.LDig ) - low ( bh.FDig ) {$else} Integer {$endif};      d :     {$ifopt r+} low ( bh.LDig ) - high ( bh.FDig )..high ( bh.LDig ) - low ( bh.FDig ) {$else} Integer {$endif};      lLdig : {$ifopt r+} low ( lFDig ) + low ( d )..high ( lFDig ) + high ( d ) {$else} Integer {$endif};      tm :    {$ifopt r+} low ( lLdig ) - high ( bh2.Singles )..high ( lLdig ) - high ( bh2.Singles ) {$else} Integer {$endif};      i2 :    {$ifopt r+} low ( lFDig )..high ( lLdig ) {$else} Integer {$endif};      i3 :    {$ifopt r+} low ( lFDig )..high ( lLdig ) {$else} Integer {$endif};      ie :    {$ifopt r+} low ( lFDig )..high ( lLdig ) {$else} Integer {$endif};      i4 :    {$ifopt r+} low ( lFDig )..high ( lLdig ) {$else} Integer {$endif};      nFDig : {$ifopt r+} low ( i2 )..high ( i2 ) {$else} Integer {$endif};      nLDig : {$ifopt r+} low ( i2 )..high ( i2 ) {$else} Integer {$endif};      dd :    {$ifopt r+} 0..9 {$else} Integer {$endif};      Add :   {$ifopt r+} 0..99 {$else} Integer {$endif};      ue :    {$ifopt r+} 0..99 {$else} Integer {$endif};      v3 :    {$ifopt r+} low ( bh.Singles[0] ) - high ( bh2.singles[9] ) * high ( dd ) - high ( ue )..high ( bh.Singles[0] ) - low ( bh2.singles[9] ) * low ( dd ) - low ( ue ) {$else} Integer {$endif};      v4 :    {$ifopt r+} low ( bh.Singles[0] ) + low ( add )..high ( bh.Singles[0] ) + high ( add ) {$else} Integer {$endif};      FlipFlop,      nz,      sf,      sh,      fdset : Boolean;{      bh1p : ARRAY [ Boolean ] of ^ tBCD_helper;}    begin{ test:      bh1p[false] := @ bh1[false];      bh1p[true] := @ bh1[true];      v := bh1[false].singles[0];      v := bh1[true].singles[0];      v := bh1p[false]^.singles[0];      v := bh1p[true]^.singles[0];      v := bh1[nz].singles[0];      v := bh1p[nz]^.singles[0];}      unpack_BCD ( Divisor, bh2 );      unpack_BCD ( Dividend, bh1[False] );      p := bh1[False].FDig - bh2.FDig;      _SELECT        _WHEN bh2.Prec = 0          _THEN RAISE eBCDException.create ( 'Division by zero' );        _WHEN bh1[False].Prec = 0          _THEN BCDout := NullBCD;        _WHEN p < low ( bh2.Singles )          _THEN RAISE eBCDOverflowException.create ( 'in BCDDivide' );        _WHENOTHER          bh := null_.bh;          bh.Neg := bh1[False].Neg XOR bh2.Neg;          if p <= high ( bh.Singles )            then begin              bh1[True] := null_.bh;              FlipFlop := False;              fdset := p > 0;              if fdset                then bh.FDig := 0;              add := 0;              nz := True;              while nz do                WITH bh1[FlipFlop] do                  begin{WriteLn('#####');dumpbh ( bh1[flipflop] );dumpbh ( bh2 );dumpbh ( bh );}                    if ( Singles[FDig] + bh2.Singles[bh2.FDig] ) = 0                      then begin                        if ( FDig >= LDig )                           OR ( bh2.FDig >= bh2.LDig )                          then nz := False                          else begin                            Inc ( FDig );                            Inc ( bh2.FDig );                           end;                       end                      else begin                        v1 := Singles[FDig];                        v2 := bh2.Singles[bh2.FDig];                        sh := v1 < v2;                        if ( v1 = v2 )                          then begin                            nz := False;                            i3 := Succ ( FDig );                            ie := LDig;                            while ( i3 <= ie ) AND ( NOT nz ) AND ( NOT sh ) do                              begin                                v1 := Singles[i3];                                v2 := bh2.Singles[i3 - p];                                if v1 <> v2                                  then begin                                    nz := True;                                    if v1 < v2                                      then sh := True;                                   end;                                Inc ( i3 );                               end;                           end;                        if NOT nz                          then Add := 1                          else begin                            if sh                              then begin                                Inc ( p );{if p > 3 then halt;}                                if p > high ( bh.Singles )                                  then nz := False                                  else Dec ( bh2.FDig );                               end                              else begin                                lFDig := FDig;                                d1 := LDig - FDig;                                d2 := bh2.LDig - bh2.FDig;                                if d1 > d2                                  then d := d1                                  else d := d2;                                lLdig := lFDig + d;                                if lLdig > high ( bh2.Singles )                                  then begin                                    tm := ( lLdig ) - high ( bh2.Singles );                                    d := d - tm;                                    lLdig := lLdig - tm;        {runden?}                                   end;                                sf := True;                                Add := 0;                                nFDig := 0;                                nLDig := 0;                                ue := 0;                                dd := Singles[lFDig] DIV ( bh2.Singles[lFDig - p] + 1 );{                                dd := 1;}                                if dd < 1                                  then dd := 1;{writeln ( 'p=', p, ' dd=', dd, ' lFdig=', lfdig, ' lldig=', lldig );}                                for i2 := lLdig DOWNTO lFDig do                                  begin                                    v3 := Singles[i2] - bh2.Singles[i2 - p] * dd - ue;                                    ue := 0;                                    while v3 < 0 do                                      begin                                        Inc ( ue );;                                        v3 := v3 + 10;                                       end;{                                    if v3 <> 0                                      then begin}                                        bh1[NOT FlipFlop].Singles[i2] := v3;{                                        nFDig := i2;                                        if sf                                          then begin                                            nLDig := i2;                                            sf := False;                                           end;                                       end;}                                   end;                                            sf := False;                                nfdig := lfdig;                                nldig := lldig;                                Inc ( Add, dd );                                if NOT fdset                                  then begin                                    bh.FDig := p;                                    fdset := True;                                   end;                                if bh.LDig < p                                  then begin                                    bh.LDig := p;                                    if ( bh.LDig - bh.FDig ) > Succ ( MaxFmtBCDFractionSize )                                      then nz := False;                                   end;                                if sf                                  then nz := False                                  else begin                                    FillChar ( bh1[FlipFlop], SizeOf ( bh1[FlipFlop] ), #0 );                                    FlipFlop := NOT FlipFlop;                                    WITH bh1[FlipFlop] do                                      begin                                        FDig := nFDig;                                        LDig := nLDig;                                       end;                                   end;                               end;                           end;                        if Add <> 0                          then begin                            i4 := p;                            while ( Add <> 0 ) AND ( i4 >= bh.FDig ) do                              begin{writeln ( '> ', i4, ' ', bh.Singles[i4], ' ', Add );}                                v4 := bh.Singles[i4] + Add;                                Add := v4 DIV 10;                                bh.Singles[i4] := v4 MOD 10;                                Dec ( i4 );                               end;                            if Add <> 0                              then begin                                Dec ( bh.FDig );                                bh.Singles[bh.FDig] := Add;                                Add := 0;                               end;                           end;                       end;                   end;             end;          WITH bh do            begin              if LDig < 0                then LDig := 0;              if LDig > 0                then Plac := LDig                else Plac := 0;             end;          if NOT pack_BCD ( bh, BCDout )            then begin              RAISE eBCDOverflowException.create ( 'in BCDDivide' );             end;       _endSELECT     end;  procedure BCDDivide ( const Dividend,                              Divisor : FmtBCDStringtype;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( StrToBCD ( Dividend ), StrToBCD ( Divisor ), BCDout );     end;  procedure BCDDivide ( const Dividend : tBCD;                        const Divisor : myRealtype;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( Dividend, DoubleToBCD ( Divisor ), BCDout );     end;  procedure BCDDivide ( const Dividend : tBCD;                        const Divisor : FmtBCDStringtype;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( Dividend, StrToBCD ( Divisor ), BCDout );     end;{ TBCD variant creation utils }  procedure VarFmtBCDCreate (   var aDest : Variant;                              const aBCD : tBCD );    begin      VarClear(aDest);      TVarData(aDest).Vtype:=FMTBcdFactory.Vartype;      TVarData(aDest).VPointer:=TFMTBcdVarData.create(aBCD);     end;  function VarFmtBCDCreate : Variant;    begin      VarFmtBCDCreate ( result, NullBCD );     end;  function VarFmtBCDCreate ( const aValue : FmtBCDStringtype;                                   Precision,                                   Scale : Word ) : Variant;    begin      VarFmtBCDCreate ( result, StrToBCD ( aValue ) );     end;  function VarFmtBCDCreate ( const aValue : myRealtype;                                   Precision : Word = 18;                                   Scale : Word = 4 ) : Variant;    begin      VarFmtBCDCreate ( result, DoubleToBCD ( aValue ) );     end;  function VarFmtBCDCreate ( const aBCD : tBCD ) : Variant;    begin      VarFmtBCDCreate ( result, aBCD );     end;  function VarIsFmtBCD ( const aValue : Variant ) : Boolean;    begin      result:=false;      not_implemented;    end;  function VarFmtBCD : TVartype;    begin      result:=0;      not_implemented;    end;  { Formatting BCD as string }  function BCDToStrF ( const BCD : tBCD;                             Format : TFloatFormat;                       const Precision,                             Digits : Integer ) : FmtBCDStringtype;    begin      not_implemented;      result:='';    end;  function FormatBCD ( const Format : string;                             BCD : tBCD ) : FmtBCDStringtype;    begin      not_implemented;      result:='';    end;{$ifdef additional_routines}  function CurrToBCD ( const Curr : currency ) : tBCD; make_Inline    begin      CurrToBCD ( Curr, result );     end;  procedure BCDAdd ( const BCDIn : tBCD;                     const IntIn : myInttype;                       var BCDout : tBCD );    var      BCD : tBCD;      bhr : tBCD_helper;      p :  {$ifopt r+} low ( bhr.FDig ) - 1..0 {$else} Integer {$endif};      ue : {$ifopt r+} 0..high ( IntIn ) - 9 {$else} Integer {$endif};      v :  {$ifopt r+} 0..{high ( ue ) + 9} high ( IntIn ) {$else} Integer {$endif};      nz : Boolean;    begin      if IntIn = 0        then begin          BCDout := BCDIn;          EXIT;         end;      if IntIn = low ( myInttype )        then begin{$if declared ( myMinIntBCD ) }          BCDAdd ( BCDIn, myMinIntBCD, BCDout );          EXIT;{$else}          RAISE eBCDOverflowException.create ( 'in BCDAdd' );{$endif}         end;      if IsBCDNegative ( BCDIn )        then begin          BCD := BCDIn;          BCDNegate ( BCD );          if IntIn < 0            then BCDAdd ( BCD, -IntIn, BCDout )            else BCDSubtract ( BCD, IntIn, BCDout );          BCDNegate ( BCDout );          EXIT;         end;      if IntIn < 0        then begin          BCDSubtract ( BCDIn, -IntIn, BCDout );          EXIT;         end;      if IntIn > ( high ( IntIn ) - 9 )        then begin          BCDAdd ( BCDIn, IntegerToBCD ( IntIn ), BCDout );          EXIT;         end;      unpack_BCD ( BCDIn, bhr );      p := 0;      nz := True;      ue := IntIn;      while nz do        begin          v := bhr.Singles[p] + ue;          bhr.Singles[p] := v MOD 10;          ue := v DIV 10;          if ue = 0            then nz := False            else Dec ( p );         end;      if p < bhr.FDig        then begin          bhr.FDig := p;          bhr.Prec := bhr.Prec + ( bhr.FDig - p );         end;      if NOT pack_BCD ( bhr, BCDout )        then begin          RAISE eBCDOverflowException.create ( 'in BCDAdd' );         end;     end;  procedure BCDSubtract ( const BCDIn : tBCD;                          const IntIn : myInttype;                            var BCDout : tBCD );{}    var      BCD : tBCD;      bhr : tBCD_helper;      p  : {$ifopt r+} low ( bhr.FDig ) - 1..0 {$else} Integer {$endif};      ue : {$ifopt r+} 0..pred ( 100000000 ) {$else} Integer {$endif};      v  : {$ifopt r+} -9..9 {$else} Integer {$endif};      direct : Boolean;{}    begin      if IntIn = 0        then begin          BCDout := BCDIn;          EXIT;         end;      if IntIn = low ( myInttype )        then begin{$if declared ( myMinIntBCD ) }          BCDSubtract ( BCDIn, myMinIntBCD, BCDout );          EXIT;{$else}          RAISE eBCDOverflowException.create ( 'in BCDSubtract' );{$endif}         end;      if IsBCDNegative ( BCDIn )        then begin          BCD := BCDIn;          BCDNegate ( BCD );          if IntIn < 0            then BCDSubtract ( BCD, -IntIn, BCDout )            else BCDAdd ( BCD, IntIn, BCDout );          BCDNegate ( BCDout );          EXIT;         end;      if IntIn < 0        then begin          BCDAdd ( BCDIn, -IntIn, BCDout );          EXIT;         end;      direct := False;      case BCDIn.Precision           -{$ifndef bigger_BCD}           ( BCDIn.SignSpecialPlaces AND PlacesMask ){$else}           BCDIn.Places{$endif}          of        2: direct := IntIn < 10;        3: direct := IntIn < 100;        4: direct := IntIn < 1000;        5: direct := IntIn < 10000;        6: direct := IntIn < 100000;        7: direct := IntIn < 1000000;        8: direct := IntIn < 10000000;        9: direct := IntIn < 100000000;       end;{write(direct);dumpbcd(bcdin);write('[',intin,']');}      if direct        then begin          unpack_BCD ( BCDIn, bhr );          WITH bhr do            begin              p := 0;              ue := IntIn;              while p >= FDig do                begin                  v := Singles[p] - ue MOD 10;                  ue := ue DIV 10;                  if v < 0                    then begin                      v := v + 10;                      ue := ue + 1;                     end;                  Singles[p] := v;                  Dec ( p );                 end;             end;          if NOT pack_BCD ( bhr, BCDout )            then begin              RAISE eBCDOverflowException.create ( 'in BCDSubtract' );             end;         end        else{}        BCDSubtract ( BCDIn, IntegerToBCD ( IntIn ), BCDout );     end;  procedure BCDAdd ( const IntIn : myInttype;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline    begin      BCDAdd ( BCDIn, IntIn, BCDout );     end;  procedure BCDAdd ( const BCDIn : tBCD;                     const DoubleIn : myRealtype;                       var BCDout : tBCD ); make_Inline    begin      BCDAdd ( BCDIn, DoubleToBCD ( DoubleIn ), BCDout );     end;  procedure BCDAdd ( const DoubleIn : myRealtype;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline    begin      BCDAdd ( DoubleToBCD ( DoubleIn ), BCDIn, BCDout );     end;  procedure BCDAdd ( const BCDIn : tBCD;                     const Currin : currency;                       var BCDout : tBCD ); make_Inline    begin      BCDAdd ( BCDIn, CurrToBCD ( Currin ), BCDout );     end;  procedure BCDAdd ( const Currin : currency;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline    begin      BCDAdd ( CurrToBCD ( Currin ), BCDIn, BCDout );     end;{$ifdef comproutines}  procedure BCDAdd ( const BCDIn : tBCD;                     const Compin : Comp;                       var BCDout : tBCD ); make_Inline    begin      BCDAdd ( BCDIn, CompToBCD ( Compin ), BCDout );     end;  procedure BCDAdd ( const Compin : Comp;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline    begin      BCDAdd ( CompToBCD ( Compin ), BCDIn, BCDout );     end;{$endif}  procedure BCDAdd ( const BCDIn : tBCD;                     const StringIn : FmtBCDStringtype;                       var BCDout : tBCD ); make_Inline    begin      BCDAdd ( BCDIn, StrToBCD ( StringIn ), BCDout );     end;  procedure BCDAdd ( const StringIn : FmtBCDStringtype;                     const BCDIn : tBCD;                       var BCDout : tBCD ); make_Inline    begin      BCDAdd ( StrToBCD ( StringIn ), BCDIn, BCDout );     end;  procedure BCDAdd ( const StringIn1,                           StringIn2 : FmtBCDStringtype;                     var BCDout : tBCD ); make_Inline    begin      BCDAdd ( StrToBCD ( StringIn1 ), StrToBCD ( StringIn2 ), BCDout );     end;  procedure BCDSubtract ( const IntIn : myInttype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( BCDIn, IntIn, BCDout );      BCDNegate ( BCDout );     end;  procedure BCDSubtract ( const BCDIn : tBCD;                          const DoubleIn : myRealtype;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( BCDIn, DoubleToBCD ( DoubleIn ), BCDout );     end;  procedure BCDSubtract ( const DoubleIn : myRealtype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( DoubleToBCD ( DoubleIn ), BCDIn, BCDout );     end;  procedure BCDSubtract ( const BCDIn : tBCD;                          const Currin : currency;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( BCDIn, CurrToBCD ( Currin ), BCDout );     end;  procedure BCDSubtract ( const Currin : currency;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( CurrToBCD ( Currin ), BCDIn, BCDout );     end;{$ifdef comproutines}  procedure BCDSubtract ( const BCDIn : tBCD;                          const Compin : Comp;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( BCDIn, CompToBCD ( Compin ), BCDout );     end;  procedure BCDSubtract ( const Compin : Comp;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( CompToBCD ( Compin ), BCDIn, BCDout );     end;{$endif}  procedure BCDSubtract ( const BCDIn : tBCD;                          const StringIn : FmtBCDStringtype;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( BCDIn, StrToBCD ( StringIn ), BCDout );     end;  procedure BCDSubtract ( const StringIn : FmtBCDStringtype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( StrToBCD ( StringIn ), BCDIn, BCDout );     end;  procedure BCDSubtract ( const StringIn1,                                StringIn2 : FmtBCDStringtype;                            var BCDout : tBCD ); make_Inline    begin      BCDSubtract ( StrToBCD ( StringIn1 ), StrToBCD ( StringIn2 ), BCDout );     end;  procedure BCDMultiply ( const BCDIn : tBCD;                          const IntIn : myInttype;                            var BCDout : tBCD );    var      bh : tBCD_helper;      bhr : tBCD_helper;      bhrr : tBCD_helper_big;      int : {$ifopt r+} 0..high ( bhrr.Singles[0] ) DIV 10 {$else} Integer {$endif};      i1 :  {$ifopt r+} low ( bh.Singles )..high ( bh.Singles ) {$else} Integer {$endif};      i3 :  {$ifopt r+} low ( bhr.Singles )..high ( bhr.Singles ) {$else} Integer {$endif};      v :   {$ifopt r+} low ( bhrr.Singles[0] ) + low ( bhrr.Singles[0] ) DIV 10..high ( bhrr.Singles[0] ) + high ( bhrr.Singles[0] ) DIV 10 {$else} Integer {$endif};      ue :  {$ifopt r+} 1 * ( low ( bhrr.Singles[0] ) + low ( bhrr.Singles[0] ) DIV 10 ) DIV 10                       ..( high ( bhrr.Singles[0] ) + high ( bhrr.Singles[0] ) DIV 10 ) DIV 10 {$else} Integer {$endif};    begin      if IntIn = 0        then begin          BCDout := NullBCD;          EXIT;         end;      if IntIn = 1        then begin          BCDout := BCDIn;          EXIT;         end;      if IntIn = -1        then begin          BCDout := BCDIn;          BCDNegate ( BCDout );          EXIT;         end;      if IntIn = low ( myInttype )        then begin{$if declared ( myMinIntBCD ) }          BCDMultiply ( BCDIn, myMinIntBCD, BCDout );          EXIT;{$else}          RAISE eBCDOverflowException.create ( 'in BCDmultiply' );{$endif}         end;      if Abs ( IntIn ) > low ( bhrr.Singles[0] ) DIV 10        then begin          BCDMultiply ( BCDIn, IntegerToBCD ( IntIn ), BCDout );          EXIT;         end;      unpack_BCD ( BCDIn, bh );      if bh.Prec = 0        then begin          BCDout := NullBCD;          EXIT;         end;      bhr := null_.bh;      bhrr := null_.bhb;      int := Abs ( IntIn );      WITH bhrr do        begin          Neg := bh.Neg XOR ( IntIn < 0 );          FDig := bh.FDig;          LDig := bh.LDig;          for i1 := bh.FDig TO bh.LDig do              Singles[i1] := bh.Singles[i1] * int;{for i3 := fdig to ldig do  write ( ' ', singles[i3] );writeln;}          ue := 0;          for i3 := LDig DOWNTO FDig do            begin              v := Singles[i3] + ue;              ue := v DIV 10;              v := v MOD 10;              bhr.Singles[i3] := v;             end;          while ue <> 0 do            begin              Dec ( FDig );              if FDig < low ( bhr.Singles )                then RAISE eBCDOverflowException.create ( 'in BCDMultiply' );              bhr.Singles[FDig] := ue MOD 10;              ue := ue DIV 10;             end;          bhr.Plac := LDig;          bhr.FDig := FDig;          if LDig > high ( bhr.Singles )            then bhr.LDig := high ( bhr.Singles )            else bhr.LDig := LDig;         end;      if NOT pack_BCD ( bhr, BCDout )        then begin          RAISE eBCDOverflowException.create ( 'in BCDMultiply' );         end;     end;  procedure BCDMultiply ( const IntIn : myInttype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( BCDIn, IntIn, BCDout );     end;  procedure BCDMultiply ( const DoubleIn : myRealtype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( DoubleToBCD ( DoubleIn ), BCDIn, BCDout );     end;  procedure BCDMultiply ( const BCDIn : tBCD;                          const Currin : currency;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( BCDIn, CurrToBCD ( Currin ), BCDout );     end;  procedure BCDMultiply ( const Currin : currency;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( CurrToBCD ( Currin ), BCDIn, BCDout );     end;{$ifdef comproutines}  procedure BCDMultiply ( const BCDIn : tBCD;                          const Compin : Comp;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( BCDIn, CompToBCD ( Compin ), BCDout );     end;  procedure BCDMultiply ( const Compin : Comp;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( CompToBCD ( Compin ), BCDIn, BCDout );     end;{$endif}  procedure BCDMultiply ( const StringIn : FmtBCDStringtype;                          const BCDIn : tBCD;                            var BCDout : tBCD ); make_Inline    begin      BCDMultiply ( StrToBCD ( StringIn ), BCDIn, BCDout );     end;  procedure BCDDivide ( const Dividend : tBCD;                        const Divisor : myInttype;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( Dividend, IntegerToBCD ( Divisor ), BCDout );     end;  procedure BCDDivide ( const Dividend : myInttype;                        const Divisor : tBCD;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( IntegerToBCD ( Dividend ), Divisor, BCDout );     end;  procedure BCDDivide ( const Dividend : myRealtype;                        const Divisor : tBCD;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( DoubleToBCD ( Dividend ), Divisor, BCDout );     end;  procedure BCDDivide ( const BCDIn : tBCD;                        const Currin : currency;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( BCDIn, CurrToBCD ( Currin ), BCDout );     end;  procedure BCDDivide ( const Currin : currency;                        const BCDIn : tBCD;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( CurrToBCD ( Currin ), BCDIn, BCDout );     end;{$ifdef comproutines}  procedure BCDDivide ( const BCDIn : tBCD;                        const Compin : Comp;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( BCDIn, CompToBCD ( Compin ), BCDout );     end;  procedure BCDDivide ( const Compin : Comp;                        const BCDIn : tBCD;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( CompToBCD ( Compin ), BCDIn, BCDout );     end;{$endif}  procedure BCDDivide ( const Dividend : FmtBCDStringtype;                        const Divisor : tBCD;                          var BCDout : tBCD ); make_Inline    begin      BCDDivide ( StrToBCD ( Dividend ), Divisor, BCDout );     end;  operator = ( const BCD1,                     BCD2 : tBCD ) z : Boolean; make_Inline    begin      z := BCDCompare ( BCD1, BCD2 ) = 0;     end;  operator < ( const BCD1,                     BCD2 : tBCD ) z : Boolean; make_Inline    begin      z := BCDCompare ( BCD1, BCD2 ) < 0;     end;  operator > ( const BCD1,                     BCD2 : tBCD ) z : Boolean; make_Inline    begin      z := BCDCompare ( BCD1, BCD2 ) > 0;     end;  operator <= ( const BCD1,                      BCD2 : tBCD ) z : Boolean; make_Inline    begin      z := BCDCompare ( BCD1, BCD2 ) <= 0;     end;  operator >= ( const BCD1,                      BCD2 : tBCD ) z : Boolean; make_Inline    begin      z := BCDCompare ( BCD1, BCD2 ) >= 0;     end;(* ########################            not allowed: why?  operator + ( const BCD : tBCD ) z : tBCD; make_Inline    begin      z := bcd;     end;##################################################### *)  operator - ( const BCD : tBCD ) z : tBCD; make_Inline    begin      z := BCD;      BCDNegate ( z );     end;  operator + ( const BCD1,                     BCD2 : tBCD ) z : tBCD; make_Inline    begin      BCDAdd ( BCD1, BCD2, z );     end;  operator + ( const BCD : tBCD;               const i : myInttype ) z : tBCD; make_Inline    begin      BCDAdd ( BCD, i, z );     end;  operator + ( const i : myInttype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDAdd ( i, BCD, z );     end;  operator + ( const BCD : tBCD;               const r : myRealtype ) z : tBCD; make_Inline    begin      BCDAdd ( BCD, DoubleToBCD ( r ), z );     end;  operator + ( const r : myRealtype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDAdd ( DoubleToBCD ( r ), BCD, z );     end;  operator + ( const BCD : tBCD;               const c : currency ) z : tBCD; make_Inline    begin      BCDAdd ( BCD, CurrToBCD ( c ), z );     end;  operator + ( const c : currency;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDAdd ( CurrToBCD ( c ), BCD, z );     end;{$ifdef comproutines}  operator + ( const BCD : tBCD;               const c : Comp ) z : tBCD; make_Inline    begin      BCDAdd ( BCD, CompToBCD ( c ), z );     end;  operator + ( const c : Comp;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDAdd ( CompToBCD ( c ), BCD, z );     end;{$endif}  operator + ( const BCD : tBCD;               const s : FmtBCDStringtype ) z : tBCD; make_Inline    begin      BCDAdd ( BCD, StrToBCD ( s ), z );     end;  operator + ( const s : FmtBCDStringtype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDAdd ( StrToBCD ( s ), BCD, z );     end;  operator - ( const BCD1,                     BCD2 : tBCD ) z : tBCD; make_Inline    begin      BCDSubtract ( BCD1, BCD2, z );     end;  operator - ( const BCD : tBCD;               const i : myInttype ) z : tBCD; make_Inline    begin      BCDSubtract ( BCD, i, z );     end;  operator - ( const i : myInttype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDSubtract ( BCD, i, z );      BCDNegate ( z );     end;  operator - ( const BCD : tBCD;               const r : myRealtype ) z : tBCD; make_Inline    begin      BCDSubtract ( BCD, DoubleToBCD ( r ), z );     end;  operator - ( const r : myRealtype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDSubtract ( DoubleToBCD ( r ), BCD, z );     end;  operator - ( const BCD : tBCD;               const c : currency ) z : tBCD; make_Inline    begin      BCDSubtract ( BCD, CurrToBCD ( c ), z );     end;  operator - ( const c : currency;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDSubtract ( CurrToBCD ( c ), BCD, z );     end;{$ifdef comproutines}  operator - ( const BCD : tBCD;               const c : Comp ) z : tBCD; make_Inline    begin      BCDSubtract ( BCD, CompToBCD ( c ), z );     end;  operator - ( const c : Comp;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDSubtract ( CompToBCD ( c ), BCD, z );     end;{$endif}  operator - ( const BCD : tBCD;               const s : FmtBCDStringtype ) z : tBCD; make_Inline    begin      BCDSubtract ( BCD, StrToBCD ( s ), z );     end;  operator - ( const s : FmtBCDStringtype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDSubtract ( StrToBCD ( s ), BCD, z );     end;  operator * ( const BCD1,                     BCD2 : tBCD ) z : tBCD; make_Inline    begin      BCDMultiply ( BCD1, BCD2, z );     end;  operator * ( const BCD : tBCD;               const i : myInttype ) z : tBCD; make_Inline    begin      BCDMultiply ( BCD, i, z );     end;  operator * ( const i : myInttype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDMultiply ( BCD, i, z );     end;  operator * ( const BCD : tBCD;               const r : myRealtype ) z : tBCD; make_Inline    begin      BCDMultiply ( BCD, DoubleToBCD ( r ), z );     end;  operator * ( const r : myRealtype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDMultiply ( DoubleToBCD ( r ), BCD, z );     end;  operator * ( const BCD : tBCD;               const c : currency ) z : tBCD; make_Inline    begin      BCDMultiply ( BCD, CurrToBCD ( c ), z );     end;  operator * ( const c : currency;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDMultiply ( CurrToBCD ( c ), BCD, z );     end;{$ifdef comproutines}  operator * ( const BCD : tBCD;               const c : Comp ) z : tBCD; make_Inline    begin      BCDMultiply ( BCD, CompToBCD ( c ), z );     end;  operator * ( const c : Comp;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDMultiply ( CompToBCD ( c ), BCD, z );     end;{$endif}  operator * ( const BCD : tBCD;               const s : FmtBCDStringtype ) z : tBCD; make_Inline    begin      BCDMultiply ( BCD, StrToBCD ( s ), z );     end;  operator * ( const s : FmtBCDStringtype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDMultiply ( StrToBCD ( s ), BCD, z );     end;  operator / ( const BCD1,                     BCD2 : tBCD ) z : tBCD; make_Inline    begin      BCDDivide ( BCD1, BCD2, z );     end;  operator / ( const BCD : tBCD;               const i : myInttype ) z : tBCD; make_Inline    begin      BCDDivide ( BCD, i, z );     end;  operator / ( const i : myInttype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDDivide ( IntegerToBCD ( i ), BCD, z );     end;  operator / ( const BCD : tBCD;               const r : myRealtype ) z : tBCD; make_Inline    begin      BCDDivide ( BCD, DoubleToBCD ( r ), z );     end;  operator / ( const r : myRealtype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDDivide ( DoubleToBCD ( r ), BCD, z );     end;  operator / ( const BCD : tBCD;               const c : currency ) z : tBCD; make_Inline    begin      BCDDivide ( BCD, CurrToBCD ( c ), z );     end;  operator / ( const c : currency;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDDivide ( CurrToBCD ( c ), BCD, z );     end;{$ifdef comproutines}  operator / ( const BCD : tBCD;               const c : Comp ) z : tBCD; make_Inline    begin      BCDDivide ( BCD, CompToBCD ( c ), z );     end;  operator / ( const c : Comp;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDDivide ( CompToBCD ( c ), BCD, z );     end;{$endif}  operator / ( const BCD : tBCD;               const s : FmtBCDStringtype ) z : tBCD; make_Inline    begin      BCDDivide ( BCD, StrToBCD ( s ), z );     end;  operator / ( const s : FmtBCDStringtype;               const BCD : tBCD ) z : tBCD; make_Inline    begin      BCDDivide ( StrToBCD ( s ), BCD, z );     end;  operator := ( const i : Byte ) z : tBCD; make_Inline    begin      z := IntegerToBCD ( myInttype ( i ) );     end;  operator := ( const BCD : tBCD ) z : Byte; make_Inline    begin      z := BCDToInteger ( BCD );     end;  operator := ( const i : Word ) z : tBCD; make_Inline    begin      z := IntegerToBCD ( myInttype ( i ) );     end;  operator := ( const BCD : tBCD ) z : Word; make_Inline    begin      z := BCDToInteger ( BCD );     end;  operator := ( const i : longword ) z : tBCD; make_Inline    begin      z := IntegerToBCD ( myInttype ( i ) );     end;  operator := ( const BCD : tBCD ) z : longword; make_Inline    begin      z := BCDToInteger ( BCD );     end;{$if declared ( qword ) }  operator := ( const i : qword ) z : tBCD; make_Inline    begin      z := IntegerToBCD ( myInttype ( i ) );     end;  operator := ( const BCD : tBCD ) z : qword; make_Inline    begin      z := BCDToInteger ( BCD );     end;{$endif}  operator := ( const i : ShortInt ) z : tBCD; make_Inline    begin      z := IntegerToBCD ( myInttype ( i ) );     end;  operator := ( const BCD : tBCD ) z : ShortInt; make_Inline    begin      z := BCDToInteger ( BCD );     end;  operator := ( const i : smallint ) z : tBCD; make_Inline    begin      z := IntegerToBCD ( myInttype ( i ) );     end;  operator := ( const BCD : tBCD ) z : smallint; make_Inline    begin      z := BCDToInteger ( BCD );     end;  operator := ( const i : LongInt ) z : tBCD; make_Inline    begin      z := IntegerToBCD ( myInttype ( i ) );     end;  operator := ( const BCD : tBCD ) z : LongInt; make_Inline    begin      z := BCDToInteger ( BCD );     end;{$if declared ( int64 ) }  operator := ( const i : int64 ) z : tBCD; make_Inline    begin      z := IntegerToBCD ( myInttype ( i ) );     end;  operator := ( const BCD : tBCD ) z : int64; make_Inline    begin      z := BCDToInteger ( BCD );     end;{$endif}  operator := ( const r : Single ) z : tBCD; make_Inline    begin      z := DoubleToBCD ( myRealtype ( r ) );     end;  operator := ( const BCD : tBCD ) z : Single; make_Inline    begin      z := BCDToDouble ( BCD );     end;  operator := ( const r : Double ) z : tBCD; make_Inline    begin      z := DoubleToBCD ( myRealtype ( r ) );     end;  operator := ( const BCD : tBCD ) z : Double; make_Inline    begin      z := BCDToDouble ( BCD );     end;{$if sizeof ( extended ) <> sizeof ( double )}  operator := ( const r : Extended ) z : tBCD; make_Inline    begin      z := DoubleToBCD ( {myRealtype (} r {)} );     end;  operator := ( const BCD : tBCD ) z : Extended; make_Inline    begin      z := BCDToDouble ( BCD );     end;{$endif}  operator := ( const c : currency ) z : tBCD; make_Inline    begin      CurrToBCD ( c, z );     end;  operator := ( const BCD : tBCD ) z : currency; make_Inline    begin      BCDToCurr ( BCD, z );     end;{$ifdef comproutines}{$undef makedirect}{$ifdef makedirect}  operator := ( const c : Comp ) z : tBCD; make_Inline    var      cc : int64 absolute c;    begin      z := IntegerToBCD ( cc );     end;{ $define version1}          { only one of these may be defined! }{ $define version2}         { version 1 produces a compiler error (with INLINE only!)}{$define version3}         { I wasn't able to reduce the problem, sorry }{$ifdef version1}  operator := ( const BCD : tBCD ) z : Comp; make_Inline    var      zz : Comp absolute z;    begin      zz := BCDToInteger ( BCD );     end;{$endif}{$ifdef version2}  operator := ( const BCD : tBCD ) z : Comp; make_Inline    var      zz : int64;      zzz : Comp absolute zz;    begin      zz := BCDToInteger ( BCD );      z := zzz;     end;{$endif}{$ifdef version3}  operator := ( const BCD : tBCD ) z : Comp; make_Inline    var      zz : record             case Boolean of               False: ( i : int64 );               True: ( c : Comp );            end;    begin      zz.i := BCDToInteger ( BCD );      z := zz.c;     end;{$endif}{$else}  operator := ( const c : Comp ) z : tBCD; make_Inline    begin      z := CompToBCD ( c );     end;  operator := ( const BCD : tBCD ) z : Comp; make_Inline    begin      z := BCDToComp ( BCD );     end;{$endif}{$endif}  operator := ( const s : string ) z : tBCD; make_Inline    begin      z := StrToBCD ( s );     end;  operator := ( const BCD : tBCD ) z : string; make_Inline    begin      z := BCDToStr ( BCD );     end;  operator := ( const s : AnsiString ) z : tBCD; make_Inline    begin      z := StrToBCD ( s );     end;  operator := ( const BCD : tBCD ) z : AnsiString; make_Inline    begin      z := BCDToStr ( BCD );     end;{$endif}constructor TFMTBcdVarData.create;  begin    inherited create;    FBcd:=NullBCD;  end;constructor TFMTBcdVarData.create(const BCD : tBCD);  begin    inherited create;    FBcd:=BCD;  end;function TFMTBcdFactory.GetInstance(const v : TVarData): tObject;  begin    result:=tObject(v.VPointer);  end;procedure TFMTBcdFactory.BinaryOp(var Left: TVarData; const Right: TVarData; const Operation: TVarOp);  begin    case Operation of      opAdd:        TFMTBcdVarData(Left.VPointer).BCD:=TFMTBcdVarData(Left.VPointer).BCD+TFMTBcdVarData(Right.VPointer).BCD;      opSubtract:        TFMTBcdVarData(Left.VPointer).BCD:=TFMTBcdVarData(Left.VPointer).BCD-TFMTBcdVarData(Right.VPointer).BCD;      opMultiply:        TFMTBcdVarData(Left.VPointer).BCD:=TFMTBcdVarData(Left.VPointer).BCD*TFMTBcdVarData(Right.VPointer).BCD;      opDivide:        TFMTBcdVarData(Left.VPointer).BCD:=TFMTBcdVarData(Left.VPointer).BCD/TFMTBcdVarData(Right.VPointer).BCD;    else      RaiseInvalidOp;    end;  end;{$if declared ( myMinIntBCD ) }(*  {$if sizeof ( integer ) = 2 }    {$ifdef BCDgr4 }  const    myMinIntBCDValue : packed array [ 1..3 ] of Char = #$32#$76#$80;    {$endif}  {$else}    {$if sizeof ( integer ) = 4 }*)      {$ifdef BCDgr9 }  const    myMinIntBCDValue : packed array [ 1..10 ] of Char = #$21#$47#$48#$36#$48;      {$endif}(*    {$else}      {$if sizeof ( integer ) = 8 }        {$ifdef BCDgr18 }  const    myMinIntBCDValue : packed array [ 1..19 ] of Char = #$92#$23#$37#$20#$36#$85#$47#$75#$80#$80;        {$endif}      {$else}        {$fatal You have an interesting integer type! Sorry, not supported}      {$endif}    {$endif}  {$endif}*){$endif}initialization  FillChar ( null_, SizeOf ( null_ ), #0 );  FillChar ( NullBCD_, SizeOf ( NullBCD_ ), #0 );  FillChar ( OneBCD_, SizeOf ( OneBCD_ ), #0 );  OneBCD_.Precision := 1;  OneBCD_.Fraction[low ( OneBCD_.Fraction )] := $10;{$if declared ( myMinIntBCD ) }  FillChar ( myMinIntBCD, SizeOf ( myMinIntBCD ), #0 );{$ifndef bigger_BCD}  myMinIntBCD.SignSpecialPlaces := NegBit;{$else}  myMinIntBCD.Negativ := True;{$endif}  {$if sizeof ( integer ) = 2 }    {$ifdef BCDgr4 }  myMinIntBCD.Precision := 5;  Move ( myMinIntBCDValue, myMinIntBCD.Fraction, SizeOf ( myMinIntBCDValue ) );    {$endif}  {$else}    {$if sizeof ( integer ) = 4 }      {$ifdef BCDgr9 }  myMinIntBCD.Precision := 10;  Move ( myMinIntBCDValue, myMinIntBCD.Fraction, SizeOf ( myMinIntBCDValue ) );      {$endif}    {$else}      {$if sizeof ( integer ) = 8 }        {$ifdef BCDgr18 }  myMinIntBCD.Precision := 19;  Move ( myMinIntBCDValue, myMinIntBCD.Fraction, SizeOf ( myMinIntBCDValue ) );        {$endif}      {$else}        {$fatal You have an interesting integer type! Sorry, not supported}      {$endif}    {$endif}  {$endif}{$endif}  FMTBcdFactory:=TFMTBcdFactory.create;finalization  FreeAndNil(FMTBcdFactory)end.
 |