| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457 | {    Copyright (c) 2003 by Florian Klaempfl    Member of the Free Pascal development team    This unit implements the code generator for the ARM    This program is free software; you can redistribute it and/or modify    it under the terms of the GNU General Public License as published by    the Free Software Foundation; either version 2 of the License, or    (at your option) any later version.    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.  See the    GNU General Public License for more details.    You should have received a copy of the GNU General Public License    along with this program; if not, write to the Free Software    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. ****************************************************************************}unit cgcpu;{$i fpcdefs.inc}  interface    uses       globtype,symtype,symdef,       cgbase,cgutils,cgobj,       aasmbase,aasmcpu,aasmtai,aasmdata,       parabase,       cpubase,cpuinfo,cg64f32,rgcpu;    type      { tbasecgarm is shared between all arm architectures }      tbasecgarm = class(tcg)        { true, if the next arithmetic operation should modify the flags }        cgsetflags : boolean;        procedure a_load_const_cgpara(list : TAsmList;size : tcgsize;a : tcgint;const paraloc : TCGPara);override;        procedure a_load_ref_cgpara(list : TAsmList;size : tcgsize;const r : treference;const paraloc : TCGPara);override;        procedure a_loadaddr_ref_cgpara(list : TAsmList;const r : treference;const paraloc : TCGPara);override;        procedure a_call_name(list : TAsmList;const s : string; weak: boolean);override;        procedure a_call_reg(list : TAsmList;reg: tregister);override;        { move instructions }        procedure a_load_reg_ref(list : TAsmList; fromsize, tosize: tcgsize; reg : tregister;const ref : treference);override;        procedure a_load_reg_reg(list : TAsmList; fromsize, tosize : tcgsize;reg1,reg2 : tregister);override;        function a_internal_load_reg_ref(list : TAsmList; fromsize, tosize: tcgsize; reg : tregister;const ref : treference):treference;        function a_internal_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister):treference;        { fpu move instructions }        procedure a_loadfpu_reg_reg(list: TAsmList; fromsize, tosize: tcgsize; reg1, reg2: tregister); override;        procedure a_loadfpu_ref_reg(list: TAsmList; fromsize, tosize: tcgsize; const ref: treference; reg: tregister); override;        procedure a_loadfpu_reg_ref(list: TAsmList; fromsize, tosize: tcgsize; reg: tregister; const ref: treference); override;        procedure a_loadfpu_ref_cgpara(list : TAsmList;size : tcgsize;const ref : treference;const paraloc : TCGPara);override;        {  comparison operations }        procedure a_cmp_const_reg_label(list : TAsmList;size : tcgsize;cmp_op : topcmp;a : tcgint;reg : tregister;          l : tasmlabel);override;        procedure a_cmp_reg_reg_label(list : TAsmList;size : tcgsize;cmp_op : topcmp;reg1,reg2 : tregister;l : tasmlabel); override;        procedure a_jmp_name(list : TAsmList;const s : string); override;        procedure a_jmp_always(list : TAsmList;l: tasmlabel); override;        procedure a_jmp_flags(list : TAsmList;const f : TResFlags;l: tasmlabel); override;        procedure g_flags2reg(list: TAsmList; size: TCgSize; const f: TResFlags; reg: TRegister); override;        procedure g_profilecode(list : TAsmList); override;        procedure g_proc_entry(list : TAsmList;localsize : longint;nostackframe:boolean);override;        procedure g_proc_exit(list : TAsmList;parasize : longint;nostackframe:boolean); override;        procedure g_maybe_got_init(list : TAsmList); override;        procedure a_loadaddr_ref_reg(list : TAsmList;const ref : treference;r : tregister);override;        procedure g_concatcopy(list : TAsmList;const source,dest : treference;len : tcgint);override;        procedure g_concatcopy_unaligned(list : TAsmList;const source,dest : treference;len : tcgint);override;        procedure g_concatcopy_move(list : TAsmList;const source,dest : treference;len : tcgint);        procedure g_concatcopy_internal(list : TAsmList;const source,dest : treference;len : tcgint;aligned : boolean);        procedure g_overflowcheck(list: TAsmList; const l: tlocation; def: tdef); override;        procedure g_overflowCheck_loc(List:TAsmList;const Loc:TLocation;def:TDef;ovloc : tlocation);override;        procedure g_save_registers(list : TAsmList);override;        procedure g_restore_registers(list : TAsmList);override;        procedure a_jmp_cond(list : TAsmList;cond : TOpCmp;l: tasmlabel);        procedure fixref(list : TAsmList;var ref : treference);        function handle_load_store(list:TAsmList;op: tasmop;oppostfix : toppostfix;reg:tregister;ref: treference):treference; virtual;        procedure g_intf_wrapper(list: TAsmList; procdef: tprocdef; const labelname: string; ioffset: longint);override;        procedure a_loadmm_reg_reg(list: TAsmList; fromsize, tosize : tcgsize;reg1, reg2: tregister;shuffle : pmmshuffle); override;        procedure a_loadmm_ref_reg(list: TAsmList; fromsize, tosize : tcgsize;const ref: treference; reg: tregister;shuffle : pmmshuffle); override;        procedure a_loadmm_reg_ref(list: TAsmList; fromsize, tosize : tcgsize;reg: tregister; const ref: treference;shuffle : pmmshuffle); override;        procedure a_loadmm_intreg_reg(list: TAsmList; fromsize, tosize : tcgsize;intreg, mmreg: tregister; shuffle: pmmshuffle); override;        procedure a_loadmm_reg_intreg(list: TAsmList; fromsize, tosize : tcgsize;mmreg, intreg: tregister; shuffle : pmmshuffle); override;        procedure a_opmm_reg_reg(list: TAsmList; Op: TOpCG; size : tcgsize;src,dst: tregister;shuffle : pmmshuffle); override;        { Transform unsupported methods into Internal errors }        procedure a_bit_scan_reg_reg(list: TAsmList; reverse: boolean; size: TCGSize; src, dst: TRegister); override;        { try to generate optimized 32 Bit multiplication, returns true if successful generated }        function try_optimized_mul32_const_reg_reg(list: TAsmList; a: tcgint; src, dst: tregister) : boolean;        { clear out potential overflow bits from 8 or 16 bit operations  }        { the upper 24/16 bits of a register after an operation          }        procedure maybeadjustresult(list: TAsmList; op: TOpCg; size: tcgsize; dst: tregister);        { mla for thumb requires that none of the registers is equal to r13/r15, this method ensures this }        procedure safe_mla(list: TAsmList;op1,op2,op3,op4 : TRegister);      end;      { tcgarm is shared between normal arm and thumb-2 }      tcgarm = class(tbasecgarm)        procedure a_op_const_reg(list : TAsmList; Op: TOpCG; size: TCGSize; a: tcgint; reg: TRegister); override;        procedure a_op_const_ref(list : TAsmList; Op: TOpCG; size: TCGSize; a: tcgint; const ref: TReference); override;        procedure a_op_reg_reg(list : TAsmList; Op: TOpCG; size: TCGSize; src, dst: TRegister); override;        procedure a_op_const_reg_reg(list: TAsmList; op: TOpCg;          size: tcgsize; a: tcgint; src, dst: tregister); override;        procedure a_op_reg_reg_reg(list: TAsmList; op: TOpCg;          size: tcgsize; src1, src2, dst: tregister); override;        procedure a_op_const_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; a: tcgint; src, dst: tregister;setflags : boolean;var ovloc : tlocation);override;        procedure a_op_reg_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; src1, src2, dst: tregister;setflags : boolean;var ovloc : tlocation);override;        procedure a_load_const_reg(list : TAsmList; size: tcgsize; a : tcgint;reg : tregister);override;        procedure a_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister);override;        procedure g_adjust_self_value(list:TAsmList;procdef: tprocdef;ioffset: tcgint); override;        {Multiply two 32-bit registers into lo and hi 32-bit registers}        procedure a_mul_reg_reg_pair(list: tasmlist; size: tcgsize; src1,src2,dstlo,dsthi: tregister); override;      end;      { normal arm cg }      tarmcgarm = class(tcgarm)        procedure init_register_allocators;override;        procedure done_register_allocators;override;      end;      { 64 bit cg for all arm flavours }      tbasecg64farm = class(tcg64f32)      end;      { tcg64farm is shared between normal arm and thumb-2 }      tcg64farm = class(tbasecg64farm)        procedure a_op64_reg_reg(list : TAsmList;op:TOpCG;size : tcgsize;regsrc,regdst : tregister64);override;        procedure a_op64_const_reg(list : TAsmList;op:TOpCG;size : tcgsize;value : int64;reg : tregister64);override;        procedure a_op64_const_reg_reg(list: TAsmList;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64);override;        procedure a_op64_reg_reg_reg(list: TAsmList;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64);override;        procedure a_op64_const_reg_reg_checkoverflow(list: TAsmList;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64;setflags : boolean;var ovloc : tlocation);override;        procedure a_op64_reg_reg_reg_checkoverflow(list: TAsmList;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64;setflags : boolean;var ovloc : tlocation);override;        procedure a_loadmm_intreg64_reg(list: TAsmList; mmsize: tcgsize; intreg: tregister64; mmreg: tregister);override;        procedure a_loadmm_reg_intreg64(list: TAsmList; mmsize: tcgsize; mmreg: tregister; intreg: tregister64);override;      end;      tarmcg64farm = class(tcg64farm)      end;      tthumbcgarm = class(tbasecgarm)        procedure init_register_allocators;override;        procedure done_register_allocators;override;        procedure g_proc_entry(list: TAsmList; localsize: longint; nostackframe: boolean);override;        procedure g_proc_exit(list : TAsmList;parasize : longint;nostackframe:boolean); override;        procedure a_op_reg_reg(list: TAsmList; Op: TOpCG; size: TCGSize; src,dst: TRegister);override;        procedure a_op_const_reg(list: TAsmList; op: TOpCg; size: tcgsize; a: tcgint; dst: tregister);override;        procedure a_op_const_reg_reg(list: TAsmList; op: TOpCg; size: tcgsize; a: tcgint; src, dst: tregister); override;        procedure g_flags2reg(list: TAsmList; size: TCgSize; const f: TResFlags; reg: TRegister); override;        procedure a_load_ref_reg(list: TAsmList; fromsize, tosize: tcgsize; const Ref: treference; reg: tregister);override;        procedure a_load_const_reg(list: TAsmList; size: tcgsize; a: tcgint; reg: tregister);override;        procedure g_adjust_self_value(list:TAsmList;procdef: tprocdef;ioffset: tcgint); override;        function handle_load_store(list: TAsmList; op: tasmop; oppostfix: toppostfix; reg: tregister; ref: treference): treference; override;        procedure g_external_wrapper(list : TAsmList; procdef : tprocdef; const externalname : string); override;      end;      tthumbcg64farm = class(tbasecg64farm)        procedure a_op64_reg_reg(list : TAsmList;op:TOpCG;size : tcgsize;regsrc,regdst : tregister64);override;        procedure a_op64_const_reg(list : TAsmList;op:TOpCG;size : tcgsize;value : int64;reg : tregister64);override;      end;      tthumb2cgarm = class(tcgarm)        procedure init_register_allocators;override;        procedure done_register_allocators;override;        procedure a_call_reg(list : TAsmList;reg: tregister);override;        procedure a_load_const_reg(list : TAsmList; size: tcgsize; a : tcgint;reg : tregister);override;        procedure a_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister);override;        procedure a_op_reg_reg(list : TAsmList; Op: TOpCG; size: TCGSize; src, dst: TRegister); override;        procedure a_op_const_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; a: tcgint; src, dst: tregister;setflags : boolean;var ovloc : tlocation);override;        procedure a_op_reg_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; src1, src2, dst: tregister;setflags : boolean;var ovloc : tlocation);override;        procedure g_flags2reg(list: TAsmList; size: TCgSize; const f: TResFlags; reg: TRegister); override;        procedure g_proc_entry(list : TAsmList;localsize : longint;nostackframe:boolean);override;        procedure g_proc_exit(list : TAsmList;parasize : longint;nostackframe:boolean); override;        function handle_load_store(list:TAsmList;op: tasmop;oppostfix : toppostfix;reg:tregister;ref: treference):treference; override;        procedure a_loadmm_reg_reg(list: TAsmList; fromsize, tosize : tcgsize;reg1, reg2: tregister;shuffle : pmmshuffle); override;        procedure a_loadmm_ref_reg(list: TAsmList; fromsize, tosize : tcgsize;const ref: treference; reg: tregister;shuffle : pmmshuffle); override;        procedure a_loadmm_reg_ref(list: TAsmList; fromsize, tosize : tcgsize;reg: tregister; const ref: treference;shuffle : pmmshuffle); override;        procedure a_loadmm_intreg_reg(list: TAsmList; fromsize, tosize : tcgsize;intreg, mmreg: tregister; shuffle: pmmshuffle); override;        procedure a_loadmm_reg_intreg(list: TAsmList; fromsize, tosize : tcgsize;mmreg, intreg: tregister; shuffle : pmmshuffle); override;      end;      tthumb2cg64farm = class(tcg64farm)        procedure a_op64_reg_reg(list : TAsmList;op:TOpCG;size : tcgsize;regsrc,regdst : tregister64);override;      end;    const      OpCmp2AsmCond : Array[topcmp] of TAsmCond = (C_NONE,C_EQ,C_GT,                           C_LT,C_GE,C_LE,C_NE,C_LS,C_CC,C_CS,C_HI);      winstackpagesize = 4096;    function get_fpu_postfix(def : tdef) : toppostfix;    procedure create_codegen;  implementation    uses       globals,verbose,systems,cutils,       aopt,aoptcpu,       fmodule,       symconst,symsym,symtable,       tgobj,       procinfo,cpupi,       paramgr;    function get_fpu_postfix(def : tdef) : toppostfix;      begin        if def.typ=floatdef then          begin            case tfloatdef(def).floattype of              s32real:                result:=PF_S;              s64real:                result:=PF_D;              s80real:                result:=PF_E;              else                internalerror(200401272);            end;          end        else          internalerror(200401271);      end;    procedure tarmcgarm.init_register_allocators;      begin        inherited init_register_allocators;        { currently, we always save R14, so we can use it }        if (target_info.system<>system_arm_darwin) then            begin              if assigned(current_procinfo) and (current_procinfo.framepointer<>NR_R11) then                rg[R_INTREGISTER]:=trgintcpu.create(R_INTREGISTER,R_SUBWHOLE,                    [RS_R0,RS_R1,RS_R2,RS_R3,RS_R12,RS_R4,RS_R5,RS_R6,RS_R7,RS_R8,                     RS_R9,RS_R10,RS_R11,RS_R14],first_int_imreg,[])              else                rg[R_INTREGISTER]:=trgintcpu.create(R_INTREGISTER,R_SUBWHOLE,                    [RS_R0,RS_R1,RS_R2,RS_R3,RS_R12,RS_R4,RS_R5,RS_R6,RS_R7,RS_R8,                     RS_R9,RS_R10,RS_R14],first_int_imreg,[])            end        else          { r7 is not available on Darwin, it's used as frame pointer (always,            for backtrace support -- also in gcc/clang -> R11 can be used).            r9 is volatile }          rg[R_INTREGISTER]:=trgintcpu.create(R_INTREGISTER,R_SUBWHOLE,              [RS_R0,RS_R1,RS_R2,RS_R3,RS_R9,RS_R12,RS_R4,RS_R5,RS_R6,RS_R8,               RS_R10,RS_R11,RS_R14],first_int_imreg,[]);        rg[R_FPUREGISTER]:=trgcpu.create(R_FPUREGISTER,R_SUBNONE,            [RS_F0,RS_F1,RS_F2,RS_F3,RS_F4,RS_F5,RS_F6,RS_F7],first_fpu_imreg,[]);        { The register allocator currently cannot deal with multiple          non-overlapping subregs per register, so we can only use          half the single precision registers for now (as sub registers of the          double precision ones). }        if current_settings.fputype=fpu_vfpv3 then          rg[R_MMREGISTER]:=trgcpu.create(R_MMREGISTER,R_SUBFD,              [RS_D0,RS_D1,RS_D2,RS_D3,RS_D4,RS_D5,RS_D6,RS_D7,               RS_D16,RS_D17,RS_D18,RS_D19,RS_D20,RS_D21,RS_D22,RS_D23,RS_D24,RS_D25,RS_D26,RS_D27,RS_D28,RS_D29,RS_D30,RS_D31,               RS_D8,RS_D9,RS_D10,RS_D11,RS_D12,RS_D13,RS_D14,RS_D15              ],first_mm_imreg,[])        else          rg[R_MMREGISTER]:=trgcpu.create(R_MMREGISTER,R_SUBFD,              [RS_D0,RS_D1,RS_D2,RS_D3,RS_D4,RS_D5,RS_D6,RS_D7,RS_D8,RS_D9,RS_D10,RS_D11,RS_D12,RS_D13,RS_D14,RS_D15],first_mm_imreg,[]);      end;    procedure tarmcgarm.done_register_allocators;      begin        rg[R_INTREGISTER].free;        rg[R_FPUREGISTER].free;        rg[R_MMREGISTER].free;        inherited done_register_allocators;      end;     procedure tcgarm.a_load_const_reg(list : TAsmList; size: tcgsize; a : tcgint;reg : tregister);       var          imm_shift : byte;          l : tasmlabel;          hr : treference;          imm1, imm2: DWord;       begin          if not(size in [OS_8,OS_S8,OS_16,OS_S16,OS_32,OS_S32]) then            internalerror(2002090902);          if is_shifter_const(a,imm_shift) then            list.concat(taicpu.op_reg_const(A_MOV,reg,a))          else if is_shifter_const(not(a),imm_shift) then            list.concat(taicpu.op_reg_const(A_MVN,reg,not(a)))          { loading of constants with mov and orr }          else if (split_into_shifter_const(a,imm1, imm2)) then            begin              list.concat(taicpu.op_reg_const(A_MOV,reg, imm1));              list.concat(taicpu.op_reg_reg_const(A_ORR,reg,reg, imm2));            end          { loading of constants with mvn and bic }          else if (split_into_shifter_const(not(a), imm1, imm2)) then            begin              list.concat(taicpu.op_reg_const(A_MVN,reg, imm1));              list.concat(taicpu.op_reg_reg_const(A_BIC,reg,reg, imm2));            end          else            begin               reference_reset(hr,4);               current_asmdata.getjumplabel(l);               cg.a_label(current_procinfo.aktlocaldata,l);               hr.symboldata:=current_procinfo.aktlocaldata.last;               current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(longint(a)));               hr.symbol:=l;               hr.base:=NR_PC;               list.concat(taicpu.op_reg_ref(A_LDR,reg,hr));            end;       end;     procedure tcgarm.a_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister);       var         oppostfix:toppostfix;         usedtmpref: treference;         tmpreg,tmpreg2 : tregister;         so : tshifterop;         dir : integer;       begin         if (TCGSize2Size[FromSize] >= TCGSize2Size[ToSize]) then           FromSize := ToSize;         case FromSize of           { signed integer registers }           OS_8:             oppostfix:=PF_B;           OS_S8:             oppostfix:=PF_SB;           OS_16:             oppostfix:=PF_H;           OS_S16:             oppostfix:=PF_SH;           OS_32,           OS_S32:             oppostfix:=PF_None;           else             InternalError(200308297);         end;         if (ref.alignment in [1,2]) and (ref.alignment<tcgsize2size[fromsize]) then           begin             if target_info.endian=endian_big then               dir:=-1             else               dir:=1;             case FromSize of               OS_16,OS_S16:                 begin                   { only complicated references need an extra loadaddr }                   if assigned(ref.symbol) or                     (ref.index<>NR_NO) or                     (ref.offset<-4095) or                     (ref.offset>4094) or                     { sometimes the compiler reused registers }                     (reg=ref.index) or                     (reg=ref.base) then                     begin                       tmpreg2:=getintregister(list,OS_INT);                       a_loadaddr_ref_reg(list,ref,tmpreg2);                       reference_reset_base(usedtmpref,tmpreg2,0,ref.alignment);                     end                   else                     usedtmpref:=ref;                   if target_info.endian=endian_big then                     inc(usedtmpref.offset,1);                   shifterop_reset(so);so.shiftmode:=SM_LSL;so.shiftimm:=8;                   tmpreg:=getintregister(list,OS_INT);                   a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,reg);                   inc(usedtmpref.offset,dir);                   if FromSize=OS_16 then                     a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg)                   else                     a_internal_load_ref_reg(list,OS_S8,OS_S8,usedtmpref,tmpreg);                   list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg,so));                 end;               OS_32,OS_S32:                 begin                   tmpreg:=getintregister(list,OS_INT);                   { only complicated references need an extra loadaddr }                   if assigned(ref.symbol) or                     (ref.index<>NR_NO) or                     (ref.offset<-4095) or                     (ref.offset>4092) or                     { sometimes the compiler reused registers }                     (reg=ref.index) or                     (reg=ref.base) then                     begin                       tmpreg2:=getintregister(list,OS_INT);                       a_loadaddr_ref_reg(list,ref,tmpreg2);                       reference_reset_base(usedtmpref,tmpreg2,0,ref.alignment);                     end                   else                     usedtmpref:=ref;                   shifterop_reset(so);so.shiftmode:=SM_LSL;                   if ref.alignment=2 then                     begin                       if target_info.endian=endian_big then                         inc(usedtmpref.offset,2);                       a_internal_load_ref_reg(list,OS_16,OS_16,usedtmpref,reg);                       inc(usedtmpref.offset,dir*2);                       a_internal_load_ref_reg(list,OS_16,OS_16,usedtmpref,tmpreg);                       so.shiftimm:=16;                       list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg,so));                     end                   else                     begin                       tmpreg2:=getintregister(list,OS_INT);                       if target_info.endian=endian_big then                         inc(usedtmpref.offset,3);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,reg);                       inc(usedtmpref.offset,dir);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                       inc(usedtmpref.offset,dir);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg2);                       so.shiftimm:=8;                       list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg,so));                       inc(usedtmpref.offset,dir);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                       so.shiftimm:=16;                       list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg2,so));                       so.shiftimm:=24;                       list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg,so));                     end;                 end               else                 handle_load_store(list,A_LDR,oppostfix,reg,ref);             end;           end         else           handle_load_store(list,A_LDR,oppostfix,reg,ref);         if (fromsize=OS_S8) and (tosize = OS_16) then           a_load_reg_reg(list,OS_16,OS_32,reg,reg);       end;    procedure tcgarm.g_adjust_self_value(list:TAsmList;procdef: tprocdef;ioffset: tcgint);      var        hsym : tsym;        href : treference;        paraloc : Pcgparalocation;        shift : byte;      begin        { calculate the parameter info for the procdef }        procdef.init_paraloc_info(callerside);        hsym:=tsym(procdef.parast.Find('self'));        if not(assigned(hsym) and          (hsym.typ=paravarsym)) then          internalerror(200305251);        paraloc:=tparavarsym(hsym).paraloc[callerside].location;        while paraloc<>nil do          with paraloc^ do            begin              case loc of                LOC_REGISTER:                  begin                    if is_shifter_const(ioffset,shift) then                      a_op_const_reg(list,OP_SUB,size,ioffset,register)                    else                      begin                        a_load_const_reg(list,OS_ADDR,ioffset,NR_R12);                        a_op_reg_reg(list,OP_SUB,size,NR_R12,register);                      end;                  end;                LOC_REFERENCE:                  begin                    { offset in the wrapper needs to be adjusted for the stored                      return address }                    reference_reset_base(href,reference.index,reference.offset+sizeof(aint),sizeof(pint));                    if is_shifter_const(ioffset,shift) then                      a_op_const_ref(list,OP_SUB,size,ioffset,href)                    else                      begin                        a_load_const_reg(list,OS_ADDR,ioffset,NR_R12);                        a_op_reg_ref(list,OP_SUB,size,NR_R12,href);                      end;                  end                else                  internalerror(200309189);              end;              paraloc:=next;            end;      end;    procedure tbasecgarm.a_load_const_cgpara(list : TAsmList;size : tcgsize;a : tcgint;const paraloc : TCGPara);      var        ref: treference;      begin        paraloc.check_simple_location;        paramanager.allocparaloc(list,paraloc.location);        case paraloc.location^.loc of          LOC_REGISTER,LOC_CREGISTER:            a_load_const_reg(list,size,a,paraloc.location^.register);          LOC_REFERENCE:            begin               reference_reset(ref,paraloc.alignment);               ref.base:=paraloc.location^.reference.index;               ref.offset:=paraloc.location^.reference.offset;               a_load_const_ref(list,size,a,ref);            end;          else            internalerror(2002081101);        end;      end;    procedure tbasecgarm.a_load_ref_cgpara(list : TAsmList;size : tcgsize;const r : treference;const paraloc : TCGPara);      var        tmpref, ref: treference;        location: pcgparalocation;        sizeleft: aint;      begin        location := paraloc.location;        tmpref := r;        sizeleft := paraloc.intsize;        while assigned(location) do          begin            paramanager.allocparaloc(list,location);            case location^.loc of              LOC_REGISTER,LOC_CREGISTER:                a_load_ref_reg(list,location^.size,location^.size,tmpref,location^.register);              LOC_REFERENCE:                begin                  reference_reset_base(ref,location^.reference.index,location^.reference.offset,paraloc.alignment);                  { doubles in softemu mode have a strange order of registers and references }                  if location^.size=OS_32 then                    g_concatcopy(list,tmpref,ref,4)                  else                    begin                      g_concatcopy(list,tmpref,ref,sizeleft);                      if assigned(location^.next) then                        internalerror(2005010710);                    end;                end;              LOC_FPUREGISTER,LOC_CFPUREGISTER:                case location^.size of                   OS_F32, OS_F64:                     a_loadfpu_ref_reg(list,location^.size,location^.size,tmpref,location^.register);                   else                     internalerror(2002072801);                end;              LOC_VOID:                begin                  // nothing to do                end;              else                internalerror(2002081103);            end;            inc(tmpref.offset,tcgsize2size[location^.size]);            dec(sizeleft,tcgsize2size[location^.size]);            location := location^.next;          end;      end;    procedure tbasecgarm.a_loadaddr_ref_cgpara(list : TAsmList;const r : treference;const paraloc : TCGPara);      var        ref: treference;        tmpreg: tregister;      begin        paraloc.check_simple_location;        paramanager.allocparaloc(list,paraloc.location);        case paraloc.location^.loc of          LOC_REGISTER,LOC_CREGISTER:            a_loadaddr_ref_reg(list,r,paraloc.location^.register);          LOC_REFERENCE:            begin              reference_reset(ref,paraloc.alignment);              ref.base := paraloc.location^.reference.index;              ref.offset := paraloc.location^.reference.offset;              tmpreg := getintregister(list,OS_ADDR);              a_loadaddr_ref_reg(list,r,tmpreg);              a_load_reg_ref(list,OS_ADDR,OS_ADDR,tmpreg,ref);            end;          else            internalerror(2002080701);        end;      end;    procedure tbasecgarm.a_call_name(list : TAsmList;const s : string; weak: boolean);      var        branchopcode: tasmop;        r : treference;        sym : TAsmSymbol;      begin        { check not really correct: should only be used for non-Thumb cpus }        if CPUARM_HAS_BLX_LABEL in cpu_capabilities[current_settings.cputype] then          branchopcode:=A_BLX        else          branchopcode:=A_BL;        if not(weak) then          sym:=current_asmdata.RefAsmSymbol(s)        else          sym:=current_asmdata.WeakRefAsmSymbol(s);        reference_reset_symbol(r,sym,0,sizeof(pint));        if (tf_pic_uses_got in target_info.flags) and           (cs_create_pic in current_settings.moduleswitches) then          begin            include(current_procinfo.flags,pi_needs_got);            r.refaddr:=addr_pic          end        else          r.refaddr:=addr_full;        list.concat(taicpu.op_ref(branchopcode,r));{        the compiler does not properly set this flag anymore in pass 1, and        for now we only need it after pass 2 (I hope) (JM)          if not(pi_do_call in current_procinfo.flags) then            internalerror(2003060703);}        include(current_procinfo.flags,pi_do_call);      end;    procedure tbasecgarm.a_call_reg(list : TAsmList;reg: tregister);      begin        { check not really correct: should only be used for non-Thumb cpus }        if not(CPUARM_HAS_BLX in cpu_capabilities[current_settings.cputype]) then          begin            list.concat(taicpu.op_reg_reg(A_MOV,NR_R14,NR_PC));            list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,reg));          end        else          list.concat(taicpu.op_reg(A_BLX, reg));{        the compiler does not properly set this flag anymore in pass 1, and        for now we only need it after pass 2 (I hope) (JM)          if not(pi_do_call in current_procinfo.flags) then            internalerror(2003060703);}        include(current_procinfo.flags,pi_do_call);      end;     procedure tcgarm.a_op_const_reg(list : TAsmList; Op: TOpCG; size: TCGSize; a: tcgint; reg: TRegister);       begin          a_op_const_reg_reg(list,op,size,a,reg,reg);       end;     procedure tcgarm.a_op_const_ref(list : TAsmList; Op: TOpCG; size: TCGSize; a: tcgint; const ref: TReference);       var         tmpreg,tmpresreg : tregister;         tmpref : treference;       begin         tmpreg:=getintregister(list,size);         tmpresreg:=getintregister(list,size);         tmpref:=a_internal_load_ref_reg(list,size,size,ref,tmpreg);         a_op_const_reg_reg(list,op,size,a,tmpreg,tmpresreg);         a_load_reg_ref(list,size,size,tmpresreg,tmpref);       end;     procedure tcgarm.a_op_reg_reg(list : TAsmList; Op: TOpCG; size: TCGSize; src, dst: TRegister);       var         so : tshifterop;       begin         if op = OP_NEG then           begin             list.concat(taicpu.op_reg_reg_const(A_RSB,dst,src,0));             maybeadjustresult(list,OP_NEG,size,dst);           end         else if op = OP_NOT then           begin             if size in [OS_8, OS_16, OS_S8, OS_S16] then               begin                 shifterop_reset(so);                 so.shiftmode:=SM_LSL;                 if size in [OS_8, OS_S8] then                   so.shiftimm:=24                 else                   so.shiftimm:=16;                 list.concat(taicpu.op_reg_reg_shifterop(A_MVN,dst,src,so));                 {Using a shift here allows this to be folded into another instruction}                 if size in [OS_S8, OS_S16] then                   so.shiftmode:=SM_ASR                 else                   so.shiftmode:=SM_LSR;                 list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,dst,so));               end             else               list.concat(taicpu.op_reg_reg(A_MVN,dst,src));           end         else           a_op_reg_reg_reg(list,op,size,src,dst,dst);       end;    const      op_reg_reg_opcg2asmop: array[TOpCG] of tasmop =        (A_NONE,A_MOV,A_ADD,A_AND,A_NONE,A_NONE,A_MUL,A_MUL,A_NONE,A_NONE,A_ORR,         A_NONE,A_NONE,A_NONE,A_SUB,A_EOR,A_NONE,A_NONE);      op_reg_opcg2asmop: array[TOpCG] of tasmop =        (A_NONE,A_MOV,A_ADD,A_AND,A_NONE,A_NONE,A_MUL,A_MUL,A_NONE,A_NONE,A_ORR,         A_ASR,A_LSL,A_LSR,A_SUB,A_EOR,A_NONE,A_ROR);      op_reg_postfix: array[TOpCG] of TOpPostfix =        (PF_None,PF_None,PF_None,PF_None,PF_None,PF_None,PF_None,PF_None,PF_None,PF_None,PF_None,         PF_None,PF_None,PF_None,PF_None,PF_None,PF_None,PF_None);    procedure tcgarm.a_op_const_reg_reg(list: TAsmList; op: TOpCg;      size: tcgsize; a: tcgint; src, dst: tregister);      var        ovloc : tlocation;      begin        a_op_const_reg_reg_checkoverflow(list,op,size,a,src,dst,false,ovloc);      end;    procedure tcgarm.a_op_reg_reg_reg(list: TAsmList; op: TOpCg;      size: tcgsize; src1, src2, dst: tregister);      var        ovloc : tlocation;      begin        a_op_reg_reg_reg_checkoverflow(list,op,size,src1,src2,dst,false,ovloc);      end;    function opshift2shiftmode(op: TOpCg): tshiftmode;      begin        case op of          OP_SHL: Result:=SM_LSL;          OP_SHR: Result:=SM_LSR;          OP_ROR: Result:=SM_ROR;          OP_ROL: Result:=SM_ROR;          OP_SAR: Result:=SM_ASR;          else internalerror(2012070501);        end      end;    function tbasecgarm.try_optimized_mul32_const_reg_reg(list: TAsmList; a: tcgint; src, dst: tregister) : boolean;      var        multiplier : dword;        power : longint;        shifterop : tshifterop;        bitsset : byte;        negative : boolean;        first : boolean;        b,        cycles : byte;        maxeffort : byte;      begin        result:=true;        cycles:=0;        negative:=a<0;        shifterop.rs:=NR_NO;        shifterop.shiftmode:=SM_LSL;        if negative then          inc(cycles);        multiplier:=dword(abs(a));        bitsset:=popcnt(multiplier and $fffffffe);        { heuristics to estimate how much instructions are reasonable to replace the mul,          this is currently based on XScale timings }        { in the simplest case, we need a mov to load the constant and a mul to carry out the          actual multiplication, this requires min. 1+4 cycles          because the first shift imm. might cause a stall and because we need more instructions          when replacing the mul we generate max. 3 instructions to replace this mul }        maxeffort:=3;        { if the constant is not a shifter op, we need either some mov/mvn/bic/or sequence or          a ldr, so generating one more operation to replace this is beneficial }        if not(is_shifter_const(dword(a),b)) and not(is_shifter_const(not(dword(a)),b)) then          inc(maxeffort);        { if the upper 5 bits are all set or clear, mul is one cycle faster }        if ((dword(a) and $f8000000)=0) or ((dword(a) and $f8000000)=$f8000000) then          dec(maxeffort);        { if the upper 17 bits are all set or clear, mul is another cycle faster }        if ((dword(a) and $ffff8000)=0) or ((dword(a) and $ffff8000)=$ffff8000) then          dec(maxeffort);        { most simple cases }        if a=1 then          a_load_reg_reg(list,OS_32,OS_32,src,dst)        else if a=0 then          a_load_const_reg(list,OS_32,0,dst)        else if a=-1 then          a_op_reg_reg(list,OP_NEG,OS_32,src,dst)        { add up ?          basically, one add is needed for each bit being set in the constant factor          however, the least significant bit is for free, it can be hidden in the initial          instruction        }        else if (bitsset+cycles<=maxeffort) and          (bitsset<=popcnt(dword(nextpowerof2(multiplier,power)-multiplier) and $fffffffe)) then          begin            first:=true;            while multiplier<>0 do              begin                shifterop.shiftimm:=BsrDWord(multiplier);                if odd(multiplier) then                  begin                    list.concat(taicpu.op_reg_reg_reg_shifterop(A_ADD,dst,src,src,shifterop));                    dec(multiplier);                  end                else                  if first then                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,shifterop))                  else                    list.concat(taicpu.op_reg_reg_reg_shifterop(A_ADD,dst,dst,src,shifterop));                first:=false;                dec(multiplier,1 shl shifterop.shiftimm);              end;            if negative then              list.concat(taicpu.op_reg_reg_const(A_RSB,dst,dst,0));          end        { subtract from the next greater power of two? }        else if popcnt(dword(nextpowerof2(multiplier,power)-multiplier) and $fffffffe)+cycles+1<=maxeffort then          begin            first:=true;            while multiplier<>0 do              begin                if first then                  begin                    multiplier:=(1 shl power)-multiplier;                    shifterop.shiftimm:=power;                  end                else                  shifterop.shiftimm:=BsrDWord(multiplier);                if odd(multiplier) then                  begin                    list.concat(taicpu.op_reg_reg_reg_shifterop(A_RSB,dst,src,src,shifterop));                    dec(multiplier);                  end                else                  if first then                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,shifterop))                  else                    begin                      list.concat(taicpu.op_reg_reg_reg_shifterop(A_SUB,dst,dst,src,shifterop));                      dec(multiplier,1 shl shifterop.shiftimm);                    end;                first:=false;              end;            if negative then              list.concat(taicpu.op_reg_reg_const(A_RSB,dst,dst,0));          end        else          result:=false;      end;    procedure tcgarm.a_op_const_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; a: tcgint; src, dst: tregister;setflags : boolean;var ovloc : tlocation);      var        shift, lsb, width : byte;        tmpreg : tregister;        so : tshifterop;        l1 : longint;        imm1, imm2: DWord;      begin        optimize_op_const(size, op, a);        case op of          OP_NONE:            begin              if src <> dst then                a_load_reg_reg(list, size, size, src, dst);              exit;            end;          OP_MOVE:            begin              a_load_const_reg(list, size, a, dst);              exit;            end;        end;        ovloc.loc:=LOC_VOID;        if {$ifopt R+}(a<>-2147483648) and{$endif} not setflags and is_shifter_const(-a,shift) then          case op of            OP_ADD:              begin                op:=OP_SUB;                a:=aint(dword(-a));              end;            OP_SUB:              begin                op:=OP_ADD;                a:=aint(dword(-a));              end          end;        if is_shifter_const(a,shift) and not(op in [OP_IMUL,OP_MUL]) then          case op of            OP_NEG,OP_NOT:              internalerror(200308281);            OP_SHL,            OP_SHR,            OP_ROL,            OP_ROR,            OP_SAR:              begin                if a>32 then                  internalerror(200308294);                shifterop_reset(so);                so.shiftmode:=opshift2shiftmode(op);                if op = OP_ROL then                  so.shiftimm:=32-a                else                  so.shiftimm:=a;                list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));              end;            else              {if (op in [OP_SUB, OP_ADD]) and                 ((a < 0) or                  (a > 4095)) then                begin                  tmpreg:=getintregister(list,size);                  list.concat(taicpu.op_reg_const(A_MOVT, tmpreg, (a shr 16) and $FFFF));                  list.concat(taicpu.op_reg_const(A_MOV, tmpreg, a and $FFFF));                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(op_reg_reg_opcg2asmop[op],dst,src,tmpreg),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))                   ));                end              else}                begin                  if cgsetflags or setflags then                    a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(setoppostfix(                    taicpu.op_reg_reg_const(op_reg_reg_opcg2asmop[op],dst,src,a),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))));                end;              if (cgsetflags or setflags) and (size in [OS_8,OS_16,OS_32]) then                begin                  ovloc.loc:=LOC_FLAGS;                  case op of                    OP_ADD:                      ovloc.resflags:=F_CS;                    OP_SUB:                      ovloc.resflags:=F_CC;                  end;                end;          end        else          begin            { there could be added some more sophisticated optimizations }            if (op in [OP_IMUL,OP_IDIV]) and (a=-1) then              a_op_reg_reg(list,OP_NEG,size,src,dst)            { we do this here instead in the peephole optimizer because              it saves us a register }            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a,l1) and not(cgsetflags or setflags) then              a_op_const_reg_reg(list,OP_SHL,size,l1,src,dst)            { for example : b=a*5 -> b=a*4+a with add instruction and shl }            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a-1,l1) and not(cgsetflags or setflags) then              begin                if l1>32 then{roozbeh does this ever happen?}                  internalerror(200308296);                shifterop_reset(so);                so.shiftmode:=SM_LSL;                so.shiftimm:=l1;                list.concat(taicpu.op_reg_reg_reg_shifterop(A_ADD,dst,src,src,so));              end            { for example : b=a*7 -> b=a*8-a with rsb instruction and shl }            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a+1,l1) and not(cgsetflags or setflags) then              begin                if l1>32 then{does this ever happen?}                  internalerror(201205181);                shifterop_reset(so);                so.shiftmode:=SM_LSL;                so.shiftimm:=l1;                list.concat(taicpu.op_reg_reg_reg_shifterop(A_RSB,dst,src,src,so));              end            else if (op in [OP_MUL,OP_IMUL]) and not(cgsetflags or setflags) and try_optimized_mul32_const_reg_reg(list,a,src,dst) then              begin                { nothing to do on success }              end            { BIC clears the specified bits, while AND keeps them, using BIC allows to use a              broader range of shifterconstants.}            else if (op = OP_AND) and is_shifter_const(not(dword(a)),shift) then              list.concat(taicpu.op_reg_reg_const(A_BIC,dst,src,not(dword(a))))            { Doing two shifts instead of two bics might allow the peephole optimizer to fold the second shift              into the following instruction}            else if (op = OP_AND) and                    is_continuous_mask(a, lsb, width) and                    ((lsb = 0) or ((lsb + width) = 32)) then              begin                shifterop_reset(so);                if (width = 16) and                   (lsb = 0) and                   (current_settings.cputype >= cpu_armv6) then                  list.concat(taicpu.op_reg_reg(A_UXTH,dst,src))                else if (width = 8) and                   (lsb = 0) and                   (current_settings.cputype >= cpu_armv6) then                  list.concat(taicpu.op_reg_reg(A_UXTB,dst,src))                else if lsb = 0 then                  begin                    so.shiftmode:=SM_LSL;                    so.shiftimm:=32-width;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                    so.shiftmode:=SM_LSR;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,dst,so));                  end                else                  begin                    so.shiftmode:=SM_LSR;                    so.shiftimm:=lsb;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                    so.shiftmode:=SM_LSL;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,dst,so));                  end;              end            else if (op = OP_AND) and split_into_shifter_const(not(dword(a)), imm1, imm2) then              begin                list.concat(taicpu.op_reg_reg_const(A_BIC,dst,src,imm1));                list.concat(taicpu.op_reg_reg_const(A_BIC,dst,dst,imm2));              end            else if (op in [OP_ADD, OP_SUB, OP_OR, OP_XOR]) and                    not(cgsetflags or setflags) and                    split_into_shifter_const(a, imm1, imm2) then              begin                list.concat(taicpu.op_reg_reg_const(op_reg_reg_opcg2asmop[op],dst,src,imm1));                list.concat(taicpu.op_reg_reg_const(op_reg_reg_opcg2asmop[op],dst,dst,imm2));              end            else              begin                tmpreg:=getintregister(list,size);                a_load_const_reg(list,size,a,tmpreg);                a_op_reg_reg_reg_checkoverflow(list,op,size,tmpreg,src,dst,setflags,ovloc);              end;          end;        maybeadjustresult(list,op,size,dst);      end;    procedure tcgarm.a_op_reg_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; src1, src2, dst: tregister;setflags : boolean;var ovloc : tlocation);      var        so : tshifterop;        tmpreg,overflowreg : tregister;        asmop : tasmop;      begin        ovloc.loc:=LOC_VOID;        case op of          OP_NEG,OP_NOT,          OP_DIV,OP_IDIV:            internalerror(200308283);          OP_SHL,          OP_SHR,          OP_SAR,          OP_ROR:            begin              if (op = OP_ROR) and not(size in [OS_32,OS_S32]) then                internalerror(2008072801);              shifterop_reset(so);              so.rs:=src1;              so.shiftmode:=opshift2shiftmode(op);              list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src2,so));            end;          OP_ROL:            begin              if not(size in [OS_32,OS_S32]) then                internalerror(2008072801);              { simulate ROL by ror'ing 32-value }              tmpreg:=getintregister(list,OS_32);              list.concat(taicpu.op_reg_reg_const(A_RSB,tmpreg,src1, 32));              shifterop_reset(so);              so.rs:=tmpreg;              so.shiftmode:=SM_ROR;              list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src2,so));            end;          OP_IMUL,          OP_MUL:            begin              if cgsetflags or setflags then                begin                  overflowreg:=getintregister(list,size);                  if op=OP_IMUL then                    asmop:=A_SMULL                  else                    asmop:=A_UMULL;                  { the arm doesn't allow that rd and rm are the same }                  if dst=src2 then                    begin                      if dst<>src1 then                        list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,src1,src2))                      else                        begin                          tmpreg:=getintregister(list,size);                          a_load_reg_reg(list,size,size,src2,dst);                          list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,tmpreg,src1));                        end;                    end                  else                    list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,src2,src1));                  a_reg_alloc(list,NR_DEFAULTFLAGS);                  if op=OP_IMUL then                    begin                      shifterop_reset(so);                      so.shiftmode:=SM_ASR;                      so.shiftimm:=31;                      list.concat(taicpu.op_reg_reg_shifterop(A_CMP,overflowreg,dst,so));                    end                  else                    list.concat(taicpu.op_reg_const(A_CMP,overflowreg,0));                   ovloc.loc:=LOC_FLAGS;                   ovloc.resflags:=F_NE;                end              else                begin                  { the arm doesn't allow that rd and rm are the same }                  if dst=src2 then                    begin                      if dst<>src1 then                        list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,src1,src2))                      else                        begin                          tmpreg:=getintregister(list,size);                          a_load_reg_reg(list,size,size,src2,dst);                          list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,tmpreg,src1));                        end;                    end                  else                    list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,src2,src1));                end;            end;          else            begin              if cgsetflags or setflags then                a_reg_alloc(list,NR_DEFAULTFLAGS);              list.concat(setoppostfix(                taicpu.op_reg_reg_reg(op_reg_reg_opcg2asmop[op],dst,src2,src1),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))));            end;        end;        maybeadjustresult(list,op,size,dst);      end;    procedure tcgarm.a_mul_reg_reg_pair(list: tasmlist; size: tcgsize; src1,src2,dstlo,dsthi: tregister);    var      asmop: tasmop;    begin      list.concat(tai_comment.create(strpnew('tcgarm.a_mul_reg_reg_pair called')));      case size of        OS_32:  asmop:=A_UMULL;        OS_S32: asmop:=A_SMULL;        else          InternalError(2014060802);      end;      { The caller might omit dstlo or dsthi, when he is not interested in it, we still        need valid registers everywhere. In case of dsthi = NR_NO we could fall back to        32x32=32 bit multiplication}      if (dstlo = NR_NO) then        dstlo:=getintregister(list,size);      if (dsthi = NR_NO) then        dsthi:=getintregister(list,size);      list.concat(taicpu.op_reg_reg_reg_reg(asmop, dstlo, dsthi, src1,src2));    end;    function tbasecgarm.handle_load_store(list:TAsmList;op: tasmop;oppostfix : toppostfix;reg:tregister;ref: treference):treference;      var        tmpreg1,tmpreg2 : tregister;        tmpref : treference;        l : tasmlabel;      begin        tmpreg1:=NR_NO;        { Be sure to have a base register }        if (ref.base=NR_NO) then          begin            if ref.shiftmode<>SM_None then              internalerror(2014020701);            ref.base:=ref.index;            ref.index:=NR_NO;          end;        { absolute symbols can't be handled directly, we've to store the symbol reference          in the text segment and access it pc relative          For now, we assume that references where base or index equals to PC are already          relative, all other references are assumed to be absolute and thus they need          to be handled extra.          A proper solution would be to change refoptions to a set and store the information          if the symbol is absolute or relative there.        }        if (assigned(ref.symbol) and            not(is_pc(ref.base)) and            not(is_pc(ref.index))           ) or           { [#xxx] isn't a valid address operand }           ((ref.base=NR_NO) and (ref.index=NR_NO)) or           (ref.offset<-4095) or           (ref.offset>4095) or           ((oppostfix in [PF_SB,PF_H,PF_SH]) and            ((ref.offset<-255) or             (ref.offset>255)            )           ) or           (((op in [A_LDF,A_STF,A_FLDS,A_FLDD,A_FSTS,A_FSTD]) or (op=A_VSTR) or (op=A_VLDR)) and            ((ref.offset<-1020) or             (ref.offset>1020) or             ((abs(ref.offset) mod 4)<>0)            )           ) or           ((GenerateThumbCode) and            (((oppostfix in [PF_SB,PF_SH]) and (ref.offset<>0)) or             ((oppostfix=PF_None) and ((ref.offset<0) or ((ref.base<>NR_STACK_POINTER_REG) and (ref.offset>124)) or               ((ref.base=NR_STACK_POINTER_REG) and (ref.offset>1020)) or ((ref.offset mod 4)<>0))) or             ((oppostfix=PF_H) and ((ref.offset<0) or (ref.offset>62) or ((ref.offset mod 2)<>0) or ((getsupreg(ref.base) in [RS_R8..RS_R15]) and (ref.offset<>0)))) or             ((oppostfix=PF_B) and ((ref.offset<0) or (ref.offset>31) or ((getsupreg(ref.base) in [RS_R8..RS_R15]) and (ref.offset<>0))))            )           ) then          begin            fixref(list,ref);          end;        if GenerateThumbCode then          begin            { certain thumb load require base and index }            if (oppostfix in [PF_SB,PF_SH]) and              (ref.base<>NR_NO) and (ref.index=NR_NO) then              begin                tmpreg1:=getintregister(list,OS_ADDR);                a_load_const_reg(list,OS_ADDR,0,tmpreg1);                ref.index:=tmpreg1;              end;            { "hi" registers cannot be used as base or index }            if (getsupreg(ref.base) in [RS_R8..RS_R12,RS_R14]) or              ((ref.base=NR_R13) and (ref.index<>NR_NO)) then              begin                tmpreg1:=getintregister(list,OS_ADDR);                a_load_reg_reg(list,OS_ADDR,OS_ADDR,ref.base,tmpreg1);                ref.base:=tmpreg1;              end;            if getsupreg(ref.index) in [RS_R8..RS_R14] then              begin                tmpreg1:=getintregister(list,OS_ADDR);                a_load_reg_reg(list,OS_ADDR,OS_ADDR,ref.index,tmpreg1);                ref.index:=tmpreg1;              end;          end;        { fold if there is base, index and offset, however, don't fold          for vfp memory instructions because we later fold the index }        if not((op in [A_FLDS,A_FLDD,A_FSTS,A_FSTD]) or (op=A_VSTR) or (op=A_VLDR)) and           (ref.base<>NR_NO) and (ref.index<>NR_NO) and (ref.offset<>0) then          begin            if tmpreg1<>NR_NO then              begin                tmpreg2:=getintregister(list,OS_ADDR);                a_op_const_reg_reg(list,OP_ADD,OS_ADDR,ref.offset,tmpreg1,tmpreg2);                tmpreg1:=tmpreg2;              end            else              begin                tmpreg1:=getintregister(list,OS_ADDR);                a_op_const_reg_reg(list,OP_ADD,OS_ADDR,ref.offset,ref.base,tmpreg1);                ref.base:=tmpreg1;              end;            ref.offset:=0;          end;        { floating point operations have only limited references          we expect here, that a base is already set }        if ((op in [A_LDF,A_STF,A_FLDS,A_FLDD,A_FSTS,A_FSTD]) or (op=A_VSTR) or (op=A_VLDR)) and (ref.index<>NR_NO) then          begin            if ref.shiftmode<>SM_none then              internalerror(200309121);            if tmpreg1<>NR_NO then              begin                if ref.base=tmpreg1 then                  begin                    if ref.signindex<0 then                      list.concat(taicpu.op_reg_reg_reg(A_SUB,tmpreg1,tmpreg1,ref.index))                    else                      list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg1,tmpreg1,ref.index));                    ref.index:=NR_NO;                  end                else                  begin                    if ref.index<>tmpreg1 then                      internalerror(200403161);                    if ref.signindex<0 then                      list.concat(taicpu.op_reg_reg_reg(A_SUB,tmpreg1,ref.base,tmpreg1))                    else                      list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg1,ref.base,tmpreg1));                    ref.base:=tmpreg1;                    ref.index:=NR_NO;                  end;              end            else              begin                tmpreg1:=getintregister(list,OS_ADDR);                list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg1,ref.base,ref.index));                ref.base:=tmpreg1;                ref.index:=NR_NO;              end;          end;        list.concat(setoppostfix(taicpu.op_reg_ref(op,reg,ref),oppostfix));        Result := ref;      end;     procedure tbasecgarm.a_load_reg_ref(list : TAsmList; fromsize, tosize: tcgsize; reg : tregister;const ref : treference);       var         oppostfix:toppostfix;         usedtmpref: treference;         tmpreg : tregister;         dir : integer;       begin         if (TCGSize2Size[FromSize] >= TCGSize2Size[ToSize]) then           FromSize := ToSize;         case ToSize of           { signed integer registers }           OS_8,           OS_S8:             oppostfix:=PF_B;           OS_16,           OS_S16:             oppostfix:=PF_H;           OS_32,           OS_S32,           { for vfp value stored in integer register }           OS_F32:             oppostfix:=PF_None;           else             InternalError(200308299);         end;         if (ref.alignment in [1,2]) and (ref.alignment<tcgsize2size[tosize]) then           begin             if target_info.endian=endian_big then               dir:=-1             else               dir:=1;             case FromSize of               OS_16,OS_S16:                 begin                   tmpreg:=getintregister(list,OS_INT);                   usedtmpref:=ref;                   if target_info.endian=endian_big then                     inc(usedtmpref.offset,1);                   usedtmpref:=a_internal_load_reg_ref(list,OS_8,OS_8,reg,usedtmpref);                   inc(usedtmpref.offset,dir);                   a_op_const_reg_reg(list,OP_SHR,OS_INT,8,reg,tmpreg);                   a_internal_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref);                 end;               OS_32,OS_S32:                 begin                   tmpreg:=getintregister(list,OS_INT);                   usedtmpref:=ref;                   if ref.alignment=2 then                     begin                       if target_info.endian=endian_big then                         inc(usedtmpref.offset,2);                       usedtmpref:=a_internal_load_reg_ref(list,OS_16,OS_16,reg,usedtmpref);                       a_op_const_reg_reg(list,OP_SHR,OS_INT,16,reg,tmpreg);                       inc(usedtmpref.offset,dir*2);                       a_internal_load_reg_ref(list,OS_16,OS_16,tmpreg,usedtmpref);                     end                   else                     begin                       if target_info.endian=endian_big then                         inc(usedtmpref.offset,3);                       usedtmpref:=a_internal_load_reg_ref(list,OS_8,OS_8,reg,usedtmpref);                       a_op_const_reg_reg(list,OP_SHR,OS_INT,8,reg,tmpreg);                       inc(usedtmpref.offset,dir);                       a_internal_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref);                       a_op_const_reg(list,OP_SHR,OS_INT,8,tmpreg);                       inc(usedtmpref.offset,dir);                       a_internal_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref);                       a_op_const_reg(list,OP_SHR,OS_INT,8,tmpreg);                       inc(usedtmpref.offset,dir);                       a_internal_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref);                     end;                 end               else                 handle_load_store(list,A_STR,oppostfix,reg,ref);             end;           end         else           handle_load_store(list,A_STR,oppostfix,reg,ref);       end;     function tbasecgarm.a_internal_load_reg_ref(list : TAsmList; fromsize, tosize: tcgsize; reg : tregister;const ref : treference):treference;       var         oppostfix:toppostfix;       begin         case ToSize of           { signed integer registers }           OS_8,           OS_S8:             oppostfix:=PF_B;           OS_16,           OS_S16:             oppostfix:=PF_H;           OS_32,           OS_S32:             oppostfix:=PF_None;           else             InternalError(2003082910);         end;         result:=handle_load_store(list,A_STR,oppostfix,reg,ref);       end;     function tbasecgarm.a_internal_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister):treference;       var         oppostfix:toppostfix;       begin         case FromSize of           { signed integer registers }           OS_8:             oppostfix:=PF_B;           OS_S8:             oppostfix:=PF_SB;           OS_16:             oppostfix:=PF_H;           OS_S16:             oppostfix:=PF_SH;           OS_32,           OS_S32:             oppostfix:=PF_None;           else             InternalError(200308291);         end;         result:=handle_load_store(list,A_LDR,oppostfix,reg,ref);       end;     procedure tbasecgarm.a_load_reg_reg(list : TAsmList; fromsize, tosize : tcgsize;reg1,reg2 : tregister);       var         so : tshifterop;       procedure do_shift(shiftmode : tshiftmode; shiftimm : byte; reg : tregister);         begin           if GenerateThumbCode then             begin               case shiftmode of                 SM_ASR:                   a_op_const_reg_reg(list,OP_SAR,OS_32,shiftimm,reg,reg2);                 SM_LSR:                   a_op_const_reg_reg(list,OP_SHR,OS_32,shiftimm,reg,reg2);                 SM_LSL:                   a_op_const_reg_reg(list,OP_SHL,OS_32,shiftimm,reg,reg2);                 else                   internalerror(2013090301);               end;             end           else             begin               so.shiftmode:=shiftmode;               so.shiftimm:=shiftimm;               list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg,so));             end;         end;       var         instr: taicpu;         conv_done: boolean;       begin         if (tcgsize2size[fromsize]>32) or (tcgsize2size[tosize]>32) or (fromsize=OS_NO) or (tosize=OS_NO) then           internalerror(2002090901);         conv_done:=false;         if tosize<>fromsize then           begin             shifterop_reset(so);             conv_done:=true;             if tcgsize2size[tosize]<=tcgsize2size[fromsize] then               fromsize:=tosize;             if current_settings.cputype<cpu_armv6 then               case fromsize of                 OS_8:                   if GenerateThumbCode then                     a_op_const_reg_reg(list,OP_AND,OS_32,$ff,reg1,reg2)                   else                     list.concat(taicpu.op_reg_reg_const(A_AND,reg2,reg1,$ff));                 OS_S8:                   begin                     do_shift(SM_LSL,24,reg1);                     if tosize=OS_16 then                       begin                         do_shift(SM_ASR,8,reg2);                         do_shift(SM_LSR,16,reg2);                       end                     else                       do_shift(SM_ASR,24,reg2);                   end;                 OS_16:                   begin                     do_shift(SM_LSL,16,reg1);                     do_shift(SM_LSR,16,reg2);                   end;                 OS_S16:                   begin                     do_shift(SM_LSL,16,reg1);                     do_shift(SM_ASR,16,reg2)                   end;                 else                   conv_done:=false;               end             else               case fromsize of                 OS_8:                   if GenerateThumbCode then                     list.concat(taicpu.op_reg_reg(A_UXTB,reg2,reg1))                   else                     list.concat(taicpu.op_reg_reg_const(A_AND,reg2,reg1,$ff));                 OS_S8:                   begin                     if tosize=OS_16 then                       begin                         so.shiftmode:=SM_ROR;                         so.shiftimm:=16;                         list.concat(taicpu.op_reg_reg_shifterop(A_SXTB16,reg2,reg1,so));                         do_shift(SM_LSR,16,reg2);                       end                     else                       list.concat(taicpu.op_reg_reg(A_SXTB,reg2,reg1));                   end;                 OS_16:                   list.concat(taicpu.op_reg_reg(A_UXTH,reg2,reg1));                 OS_S16:                   list.concat(taicpu.op_reg_reg(A_SXTH,reg2,reg1));                 else                   conv_done:=false;               end           end;         if not conv_done and (reg1<>reg2) then           begin             { same size, only a register mov required }             instr:=taicpu.op_reg_reg(A_MOV,reg2,reg1);             list.Concat(instr);             { Notify the register allocator that we have written a move instruction so               it can try to eliminate it. }             add_move_instruction(instr);           end;       end;    procedure tbasecgarm.a_loadfpu_ref_cgpara(list : TAsmList;size : tcgsize;const ref : treference;const paraloc : TCGPara);      var         href,href2 : treference;         hloc : pcgparalocation;      begin        href:=ref;        hloc:=paraloc.location;        while assigned(hloc) do          begin            case hloc^.loc of              LOC_FPUREGISTER,LOC_CFPUREGISTER:                begin                  paramanager.allocparaloc(list,paraloc.location);                  a_loadfpu_ref_reg(list,size,size,ref,hloc^.register);                end;              LOC_REGISTER :                case hloc^.size of                  OS_32,                  OS_F32:                    begin                      paramanager.allocparaloc(list,paraloc.location);                      a_load_ref_reg(list,OS_32,OS_32,href,hloc^.register);                    end;                  OS_64,                  OS_F64:                    cg64.a_load64_ref_cgpara(list,href,paraloc);                  else                    a_load_ref_reg(list,hloc^.size,hloc^.size,href,hloc^.register);                end;              LOC_REFERENCE :                begin                  reference_reset_base(href2,hloc^.reference.index,hloc^.reference.offset,paraloc.alignment);                  { concatcopy should choose the best way to copy the data }                  g_concatcopy(list,href,href2,tcgsize2size[hloc^.size]);                end;              else                internalerror(200408241);           end;           inc(href.offset,tcgsize2size[hloc^.size]);           hloc:=hloc^.next;         end;      end;     procedure tbasecgarm.a_loadfpu_reg_reg(list: TAsmList; fromsize,tosize: tcgsize; reg1, reg2: tregister);       begin         list.concat(setoppostfix(taicpu.op_reg_reg(A_MVF,reg2,reg1),cgsize2fpuoppostfix[tosize]));       end;     procedure tbasecgarm.a_loadfpu_ref_reg(list: TAsmList; fromsize,tosize: tcgsize; const ref: treference; reg: tregister);       var         oppostfix:toppostfix;       begin         case fromsize of           OS_32,           OS_F32:             oppostfix:=PF_S;           OS_64,           OS_F64:             oppostfix:=PF_D;           OS_F80:             oppostfix:=PF_E;           else             InternalError(200309021);         end;         handle_load_store(list,A_LDF,oppostfix,reg,ref);         if fromsize<>tosize then           a_loadfpu_reg_reg(list,fromsize,tosize,reg,reg);       end;     procedure tbasecgarm.a_loadfpu_reg_ref(list: TAsmList; fromsize, tosize: tcgsize; reg: tregister; const ref: treference);       var         oppostfix:toppostfix;       begin         case tosize of           OS_F32:             oppostfix:=PF_S;           OS_F64:             oppostfix:=PF_D;           OS_F80:             oppostfix:=PF_E;           else             InternalError(200309022);         end;         handle_load_store(list,A_STF,oppostfix,reg,ref);       end;    {  comparison operations }    procedure tbasecgarm.a_cmp_const_reg_label(list : TAsmList;size : tcgsize;cmp_op : topcmp;a : tcgint;reg : tregister;      l : tasmlabel);      var        tmpreg : tregister;        b : byte;      begin        a_reg_alloc(list,NR_DEFAULTFLAGS);        if (not(GenerateThumbCode) and is_shifter_const(a,b)) or          ((GenerateThumbCode) and is_thumb_imm(a)) then          list.concat(taicpu.op_reg_const(A_CMP,reg,a))        { CMN reg,0 and CMN reg,$80000000 are different from CMP reg,$ffffffff          and CMP reg,$7fffffff regarding the flags according to the ARM manual }        else if (a<>$7fffffff) and (a<>-1) and not(GenerateThumbCode) and is_shifter_const(-a,b) then          list.concat(taicpu.op_reg_const(A_CMN,reg,-a))        else          begin            tmpreg:=getintregister(list,size);            a_load_const_reg(list,size,a,tmpreg);            list.concat(taicpu.op_reg_reg(A_CMP,reg,tmpreg));          end;        a_jmp_cond(list,cmp_op,l);        a_reg_dealloc(list,NR_DEFAULTFLAGS);      end;    procedure tbasecgarm.a_bit_scan_reg_reg(list: TAsmList; reverse: boolean; size: TCGSize; src, dst: TRegister);      begin        if reverse then          begin            list.Concat(taicpu.op_reg_reg(A_CLZ,dst,src));            list.Concat(taicpu.op_reg_reg_const(A_RSB,dst,dst,31));            list.Concat(taicpu.op_reg_reg_const(A_AND,dst,dst,255));          end        { it is decided during the compilation of the system unit if this code is used or not          so no additional check for rbit is needed                                           }        else          begin            list.Concat(taicpu.op_reg_reg(A_RBIT,dst,src));            list.Concat(taicpu.op_reg_reg(A_CLZ,dst,dst));            a_reg_alloc(list,NR_DEFAULTFLAGS);            list.Concat(taicpu.op_reg_const(A_CMP,dst,32));            if GenerateThumb2Code then              list.Concat(taicpu.op_cond(A_IT, C_EQ));            list.Concat(setcondition(taicpu.op_reg_const(A_MOV,dst,$ff),C_EQ));            a_reg_dealloc(list,NR_DEFAULTFLAGS);          end;      end;    procedure tbasecgarm.a_cmp_reg_reg_label(list : TAsmList;size : tcgsize;cmp_op : topcmp;reg1,reg2 : tregister;l : tasmlabel);      begin        a_reg_alloc(list,NR_DEFAULTFLAGS);        list.concat(taicpu.op_reg_reg(A_CMP,reg2,reg1));        a_jmp_cond(list,cmp_op,l);        a_reg_dealloc(list,NR_DEFAULTFLAGS);      end;    procedure tbasecgarm.a_jmp_name(list : TAsmList;const s : string);      var        ai : taicpu;      begin        { generate far jump, leave it to the optimizer to get rid of it }        if GenerateThumbCode then          ai:=taicpu.op_sym(A_BL,current_asmdata.RefAsmSymbol(s))        else          ai:=taicpu.op_sym(A_B,current_asmdata.RefAsmSymbol(s));        ai.is_jmp:=true;        list.concat(ai);      end;    procedure tbasecgarm.a_jmp_always(list : TAsmList;l: tasmlabel);      var        ai : taicpu;      begin        { generate far jump, leave it to the optimizer to get rid of it }        if GenerateThumbCode then          ai:=taicpu.op_sym(A_BL,l)        else          ai:=taicpu.op_sym(A_B,l);        ai.is_jmp:=true;        list.concat(ai);      end;    procedure tbasecgarm.a_jmp_flags(list : TAsmList;const f : TResFlags;l: tasmlabel);      var        ai : taicpu;        inv_flags : TResFlags;        hlabel : TAsmLabel;      begin        if GenerateThumbCode then          begin            inv_flags:=f;            inverse_flags(inv_flags);            { the optimizer has to fix this if jump range is sufficient short }            current_asmdata.getjumplabel(hlabel);            ai:=setcondition(taicpu.op_sym(A_B,hlabel),flags_to_cond(inv_flags));            ai.is_jmp:=true;            list.concat(ai);            a_jmp_always(list,l);            a_label(list,hlabel);          end        else          begin            ai:=setcondition(taicpu.op_sym(A_B,l),flags_to_cond(f));            ai.is_jmp:=true;            list.concat(ai);          end;      end;    procedure tbasecgarm.g_flags2reg(list: TAsmList; size: TCgSize; const f: TResFlags; reg: TRegister);      begin        list.concat(setcondition(taicpu.op_reg_const(A_MOV,reg,1),flags_to_cond(f)));        list.concat(setcondition(taicpu.op_reg_const(A_MOV,reg,0),inverse_cond(flags_to_cond(f))));      end;    procedure tbasecgarm.g_profilecode(list : TAsmList);      begin        if target_info.system = system_arm_linux then          begin            list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,[RS_R14]));            a_call_name(list,'__gnu_mcount_nc',false);          end        else          internalerror(2014091201);      end;    procedure tbasecgarm.g_proc_entry(list : TAsmList;localsize : longint;nostackframe:boolean);      var         ref : treference;         shift : byte;         firstfloatreg,lastfloatreg,         r : byte;         mmregs,         regs, saveregs : tcpuregisterset;         registerarea,         r7offset,         stackmisalignment : pint;         postfix: toppostfix;         imm1, imm2: DWord;         stack_parameters : Boolean;      begin        LocalSize:=align(LocalSize,4);        stack_parameters:=current_procinfo.procdef.stack_tainting_parameter(calleeside);        { call instruction does not put anything on the stack }        registerarea:=0;        tarmprocinfo(current_procinfo).stackpaddingreg:=High(TSuperRegister);        lastfloatreg:=RS_NO;        if not(nostackframe) then          begin            firstfloatreg:=RS_NO;            mmregs:=[];            case current_settings.fputype of              fpu_fpa,              fpu_fpa10,              fpu_fpa11:                begin                  { save floating point registers? }                  regs:=rg[R_FPUREGISTER].used_in_proc-paramanager.get_volatile_registers_fpu(pocall_stdcall);                  for r:=RS_F0 to RS_F7 do                    if r in regs then                      begin                        if firstfloatreg=RS_NO then                          firstfloatreg:=r;                        lastfloatreg:=r;                        inc(registerarea,12);                      end;                end;              fpu_vfpv2,              fpu_vfpv3,              fpu_vfpv3_d16:                begin;                  mmregs:=rg[R_MMREGISTER].used_in_proc-paramanager.get_volatile_registers_mm(pocall_stdcall);                end;            end;            a_reg_alloc(list,NR_STACK_POINTER_REG);            if current_procinfo.framepointer<>NR_STACK_POINTER_REG then              a_reg_alloc(list,NR_FRAME_POINTER_REG);            { save int registers }            reference_reset(ref,4);            ref.index:=NR_STACK_POINTER_REG;            ref.addressmode:=AM_PREINDEXED;            regs:=rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall);            if not(target_info.system in systems_darwin) then              begin                a_reg_alloc(list,NR_STACK_POINTER_REG);                if current_procinfo.framepointer<>NR_STACK_POINTER_REG then                  begin                    a_reg_alloc(list,NR_R12);                    list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_STACK_POINTER_REG));                  end;                { the (old) ARM APCS requires saving both the stack pointer (to                  crawl the stack) and the PC (to identify the function this                  stack frame belongs to) -> also save R12 (= copy of R13 on entry)                  and R15 -- still needs updating for EABI and Darwin, they don't                  need that }                if current_procinfo.framepointer<>NR_STACK_POINTER_REG then                  regs:=regs+[RS_FRAME_POINTER_REG,RS_R12,RS_R14,RS_R15]                else                  if (regs<>[]) or (pi_do_call in current_procinfo.flags) then                    include(regs,RS_R14);                if regs<>[] then                   begin                     for r:=RS_R0 to RS_R15 do                       if r in regs then                         inc(registerarea,4);                     { if the stack is not 8 byte aligned, try to add an extra register,                       so we can avoid the extra sub/add ...,#4 later (KB) }                     if ((registerarea mod current_settings.alignment.localalignmax) <> 0) then                       for r:=RS_R3 downto RS_R0 do                         if not(r in regs) then                           begin                             regs:=regs+[r];                             inc(registerarea,4);                             tarmprocinfo(current_procinfo).stackpaddingreg:=r;                             break;                           end;                     list.concat(setoppostfix(taicpu.op_ref_regset(A_STM,ref,R_INTREGISTER,R_SUBWHOLE,regs),PF_FD));                   end;                if current_procinfo.framepointer<>NR_STACK_POINTER_REG then                  begin                    { the framepointer now points to the saved R15, so the saved                      framepointer is at R11-12 (for get_caller_frame) }                    list.concat(taicpu.op_reg_reg_const(A_SUB,NR_FRAME_POINTER_REG,NR_R12,4));                    a_reg_dealloc(list,NR_R12);                  end;              end            else              begin                { always save r14 if we use r7 as the framepointer, because                  the parameter offsets are hardcoded in advance and always                  assume that r14 sits on the stack right behind the saved r7                }                if current_procinfo.framepointer=NR_FRAME_POINTER_REG then                  include(regs,RS_FRAME_POINTER_REG);                if (regs<>[]) or (pi_do_call in current_procinfo.flags) then                    include(regs,RS_R14);                if regs<>[] then                  begin                    { on Darwin, you first have to save [r4-r7,lr], and then                      [r8,r10,r11] and make r7 point to the previously saved                      r7 so that you can perform a stack crawl based on it                      ([r7] is previous stack frame, [r7+4] is return address                    }                    include(regs,RS_FRAME_POINTER_REG);                    saveregs:=regs-[RS_R8,RS_R10,RS_R11];                    r7offset:=0;                    for r:=RS_R0 to RS_R15 do                      if r in saveregs then                        begin                          inc(registerarea,4);                          if r<RS_FRAME_POINTER_REG then                            inc(r7offset,4);                        end;                    { save the registers }                    list.concat(setoppostfix(taicpu.op_ref_regset(A_STM,ref,R_INTREGISTER,R_SUBWHOLE,saveregs),PF_FD));                    { make r7 point to the saved r7 (regardless of whether this                      frame uses the framepointer, for backtrace purposes) }                    if r7offset<>0 then                      list.concat(taicpu.op_reg_reg_const(A_ADD,NR_FRAME_POINTER_REG,NR_R13,r7offset))                    else                      list.concat(taicpu.op_reg_reg(A_MOV,NR_R7,NR_R13));                    { now save the rest (if any) }                    saveregs:=regs-saveregs;                    if saveregs<>[] then                      begin                        for r:=RS_R8 to RS_R11 do                          if r in saveregs then                            inc(registerarea,4);                        list.concat(setoppostfix(taicpu.op_ref_regset(A_STM,ref,R_INTREGISTER,R_SUBWHOLE,saveregs),PF_FD));                      end;                  end;              end;            stackmisalignment:=registerarea mod current_settings.alignment.localalignmax;            if (LocalSize<>0) or               ((stackmisalignment<>0) and                ((pi_do_call in current_procinfo.flags) or                 (po_assembler in current_procinfo.procdef.procoptions))) then              begin                localsize:=align(localsize+stackmisalignment,current_settings.alignment.localalignmax)-stackmisalignment;                if stack_parameters and (pi_estimatestacksize in current_procinfo.flags) then                  begin                    if localsize>tarmprocinfo(current_procinfo).stackframesize then                      internalerror(2014030901)                    else                      localsize:=tarmprocinfo(current_procinfo).stackframesize-registerarea;                  end;                if is_shifter_const(localsize,shift) then                  begin                    a_reg_dealloc(list,NR_R12);                    list.concat(taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,LocalSize));                  end                else if split_into_shifter_const(localsize, imm1, imm2) then                  begin                    a_reg_dealloc(list,NR_R12);                    list.concat(taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,imm1));                    list.concat(taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,imm2));                  end                else                  begin                    if current_procinfo.framepointer=NR_STACK_POINTER_REG then                      a_reg_alloc(list,NR_R12);                    a_load_const_reg(list,OS_ADDR,LocalSize,NR_R12);                    list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R12));                    a_reg_dealloc(list,NR_R12);                  end;              end;            if (mmregs<>[]) or               (firstfloatreg<>RS_NO) then             begin               reference_reset(ref,4);               if (tg.direction*tarmprocinfo(current_procinfo).floatregstart>=1023) or                  (current_settings.fputype in [fpu_vfpv2,fpu_vfpv3,fpu_vfpv3_d16]) then                 begin                   if not is_shifter_const(tarmprocinfo(current_procinfo).floatregstart,shift) then                     begin                       a_reg_alloc(list,NR_R12);                       a_load_const_reg(list,OS_ADDR,-tarmprocinfo(current_procinfo).floatregstart,NR_R12);                       list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_R12,current_procinfo.framepointer,NR_R12));                       a_reg_dealloc(list,NR_R12);                     end                   else                     list.concat(taicpu.op_reg_reg_const(A_SUB,NR_R12,current_procinfo.framepointer,-tarmprocinfo(current_procinfo).floatregstart));                   ref.base:=NR_R12;                 end               else                 begin                   ref.base:=current_procinfo.framepointer;                   ref.offset:=tarmprocinfo(current_procinfo).floatregstart;                 end;               case current_settings.fputype of                 fpu_fpa,                 fpu_fpa10,                 fpu_fpa11:                   begin                     list.concat(taicpu.op_reg_const_ref(A_SFM,newreg(R_FPUREGISTER,firstfloatreg,R_SUBWHOLE),                       lastfloatreg-firstfloatreg+1,ref));                   end;                 fpu_vfpv2,                 fpu_vfpv3,                 fpu_vfpv3_d16:                   begin                     ref.index:=ref.base;                     ref.base:=NR_NO;                     { FSTMX is deprecated on ARMv6 and later }                     if (current_settings.cputype<cpu_armv6) then                       postfix:=PF_IAX                     else                       postfix:=PF_IAD;                     list.concat(setoppostfix(taicpu.op_ref_regset(A_FSTM,ref,R_MMREGISTER,R_SUBFD,mmregs),postfix));                   end;               end;             end;        end;      end;    procedure tbasecgarm.g_proc_exit(list : TAsmList;parasize : longint;nostackframe:boolean);      var         ref : treference;         LocalSize : longint;         firstfloatreg,lastfloatreg,         r,         shift : byte;         mmregs,         saveregs,         regs : tcpuregisterset;         registerarea,         stackmisalignment: pint;         paddingreg: TSuperRegister;         mmpostfix: toppostfix;         imm1, imm2: DWord;      begin        if not(nostackframe) then          begin            registerarea:=0;            firstfloatreg:=RS_NO;            lastfloatreg:=RS_NO;            mmregs:=[];            saveregs:=[];            case current_settings.fputype of              fpu_fpa,              fpu_fpa10,              fpu_fpa11:                begin                  { restore floating point registers? }                  regs:=rg[R_FPUREGISTER].used_in_proc-paramanager.get_volatile_registers_fpu(pocall_stdcall);                  for r:=RS_F0 to RS_F7 do                    if r in regs then                      begin                        if firstfloatreg=RS_NO then                          firstfloatreg:=r;                        lastfloatreg:=r;                        { floating point register space is already included in                          localsize below by calc_stackframe_size                         inc(registerarea,12);                        }                      end;                end;              fpu_vfpv2,              fpu_vfpv3,              fpu_vfpv3_d16:                begin;                  { restore vfp registers? }                  mmregs:=rg[R_MMREGISTER].used_in_proc-paramanager.get_volatile_registers_mm(pocall_stdcall);                end;            end;            if (firstfloatreg<>RS_NO) or               (mmregs<>[]) then              begin                reference_reset(ref,4);                if (tg.direction*tarmprocinfo(current_procinfo).floatregstart>=1023) or                   (current_settings.fputype in [fpu_vfpv2,fpu_vfpv3,fpu_vfpv3_d16]) then                  begin                    if not is_shifter_const(tarmprocinfo(current_procinfo).floatregstart,shift) then                      begin                        a_reg_alloc(list,NR_R12);                        a_load_const_reg(list,OS_ADDR,-tarmprocinfo(current_procinfo).floatregstart,NR_R12);                        list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_R12,current_procinfo.framepointer,NR_R12));                        a_reg_dealloc(list,NR_R12);                      end                    else                      list.concat(taicpu.op_reg_reg_const(A_SUB,NR_R12,current_procinfo.framepointer,-tarmprocinfo(current_procinfo).floatregstart));                    ref.base:=NR_R12;                  end                else                  begin                    ref.base:=current_procinfo.framepointer;                    ref.offset:=tarmprocinfo(current_procinfo).floatregstart;                  end;                case current_settings.fputype of                  fpu_fpa,                  fpu_fpa10,                  fpu_fpa11:                    begin                      list.concat(taicpu.op_reg_const_ref(A_LFM,newreg(R_FPUREGISTER,firstfloatreg,R_SUBWHOLE),                        lastfloatreg-firstfloatreg+1,ref));                    end;                  fpu_vfpv2,                  fpu_vfpv3,                  fpu_vfpv3_d16:                    begin                      ref.index:=ref.base;                      ref.base:=NR_NO;                      { FLDMX is deprecated on ARMv6 and later }                      if (current_settings.cputype<cpu_armv6) then                        mmpostfix:=PF_IAX                      else                        mmpostfix:=PF_IAD;                      list.concat(setoppostfix(taicpu.op_ref_regset(A_FLDM,ref,R_MMREGISTER,R_SUBFD,mmregs),mmpostfix));                    end;                end;              end;            regs:=rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall);            if (pi_do_call in current_procinfo.flags) or               (regs<>[]) or               ((target_info.system in systems_darwin) and                (current_procinfo.framepointer<>NR_STACK_POINTER_REG)) then              begin                exclude(regs,RS_R14);                include(regs,RS_R15);                if (target_info.system in systems_darwin) then                  include(regs,RS_FRAME_POINTER_REG);              end;            if not(target_info.system in systems_darwin) then              begin                { restore saved stack pointer to SP (R13) and saved lr to PC (R15).                  The saved PC came after that but is discarded, since we restore                  the stack pointer }                if (current_procinfo.framepointer<>NR_STACK_POINTER_REG) then                  regs:=regs+[RS_FRAME_POINTER_REG,RS_R13,RS_R15];              end            else              begin                { restore R8-R11 already if necessary (they've been stored                  before the others) }                saveregs:=regs*[RS_R8,RS_R10,RS_R11];                if saveregs<>[] then                  begin                    reference_reset(ref,4);                    ref.index:=NR_STACK_POINTER_REG;                    ref.addressmode:=AM_PREINDEXED;                    for r:=RS_R8 to RS_R11 do                      if r in saveregs then                        inc(registerarea,4);                    regs:=regs-saveregs;                  end;              end;            for r:=RS_R0 to RS_R15 do              if r in regs then                inc(registerarea,4);            { reapply the stack padding reg, in case there was one, see the complimentary              comment in g_proc_entry() (KB) }            paddingreg:=tarmprocinfo(current_procinfo).stackpaddingreg;            if paddingreg < RS_R4 then              if paddingreg in regs then                internalerror(201306190)              else                begin                  regs:=regs+[paddingreg];                  inc(registerarea,4);                end;            stackmisalignment:=registerarea mod current_settings.alignment.localalignmax;            if (current_procinfo.framepointer=NR_STACK_POINTER_REG) or               (target_info.system in systems_darwin) then              begin                LocalSize:=current_procinfo.calc_stackframe_size;                if (LocalSize<>0) or                   ((stackmisalignment<>0) and                    ((pi_do_call in current_procinfo.flags) or                     (po_assembler in current_procinfo.procdef.procoptions))) then                  begin                    if pi_estimatestacksize in current_procinfo.flags then                      LocalSize:=tarmprocinfo(current_procinfo).stackframesize-registerarea                    else                      localsize:=align(localsize+stackmisalignment,current_settings.alignment.localalignmax)-stackmisalignment;                    if is_shifter_const(LocalSize,shift) then                      list.concat(taicpu.op_reg_reg_const(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,LocalSize))                    else if split_into_shifter_const(localsize, imm1, imm2) then                      begin                        list.concat(taicpu.op_reg_reg_const(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,imm1));                        list.concat(taicpu.op_reg_reg_const(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,imm2));                      end                    else                      begin                        a_reg_alloc(list,NR_R12);                        a_load_const_reg(list,OS_ADDR,LocalSize,NR_R12);                        list.concat(taicpu.op_reg_reg_reg(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R12));                        a_reg_dealloc(list,NR_R12);                      end;                  end;                if (target_info.system in systems_darwin) and                   (saveregs<>[]) then                  list.concat(setoppostfix(taicpu.op_ref_regset(A_LDM,ref,R_INTREGISTER,R_SUBWHOLE,saveregs),PF_FD));                if regs=[] then                  begin                    if not(CPUARM_HAS_BX in cpu_capabilities[current_settings.cputype]) then                      list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R14))                    else                      list.concat(taicpu.op_reg(A_BX,NR_R14))                  end                else                  begin                    reference_reset(ref,4);                    ref.index:=NR_STACK_POINTER_REG;                    ref.addressmode:=AM_PREINDEXED;                    list.concat(setoppostfix(taicpu.op_ref_regset(A_LDM,ref,R_INTREGISTER,R_SUBWHOLE,regs),PF_FD));                  end;              end            else              begin                { restore int registers and return }                reference_reset(ref,4);                ref.index:=NR_FRAME_POINTER_REG;                list.concat(setoppostfix(taicpu.op_ref_regset(A_LDM,ref,R_INTREGISTER,R_SUBWHOLE,regs),PF_EA));              end;          end        else if not(CPUARM_HAS_BX in cpu_capabilities[current_settings.cputype]) then          list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R14))        else          list.concat(taicpu.op_reg(A_BX,NR_R14))      end;    procedure tbasecgarm.g_maybe_got_init(list : TAsmList);      var        ref : treference;        l : TAsmLabel;      begin        if (cs_create_pic in current_settings.moduleswitches) and           (pi_needs_got in current_procinfo.flags) and           (tf_pic_uses_got in target_info.flags) then          begin            reference_reset(ref,4);            current_asmdata.getdatalabel(l);            cg.a_label(current_procinfo.aktlocaldata,l);            ref.symbol:=l;            ref.base:=NR_PC;            ref.symboldata:=current_procinfo.aktlocaldata.last;            list.concat(Taicpu.op_reg_ref(A_LDR,current_procinfo.got,ref));            current_asmdata.getaddrlabel(l);            current_procinfo.aktlocaldata.concat(tai_const.Create_rel_sym_offset(aitconst_32bit,l,current_asmdata.RefAsmSymbol('_GLOBAL_OFFSET_TABLE_'),-8));            cg.a_label(list,l);            list.concat(Taicpu.op_reg_reg_reg(A_ADD,current_procinfo.got,NR_PC,current_procinfo.got));          end;      end;    procedure tbasecgarm.a_loadaddr_ref_reg(list : TAsmList;const ref : treference;r : tregister);      var        b : byte;        tmpref : treference;        instr : taicpu;      begin        if ref.addressmode<>AM_OFFSET then          internalerror(200309071);        tmpref:=ref;        { Be sure to have a base register }        if (tmpref.base=NR_NO) then          begin            if tmpref.shiftmode<>SM_None then              internalerror(2014020702);            if tmpref.signindex<0 then              internalerror(200312023);            tmpref.base:=tmpref.index;            tmpref.index:=NR_NO;          end;        if assigned(tmpref.symbol) or           not((is_shifter_const(tmpref.offset,b)) or               (is_shifter_const(-tmpref.offset,b))              ) then          fixref(list,tmpref);        { expect a base here if there is an index }        if (tmpref.base=NR_NO) and (tmpref.index<>NR_NO) then          internalerror(200312022);        if tmpref.index<>NR_NO then          begin            if tmpref.shiftmode<>SM_None then              internalerror(200312021);            if tmpref.signindex<0 then              a_op_reg_reg_reg(list,OP_SUB,OS_ADDR,tmpref.base,tmpref.index,r)            else              a_op_reg_reg_reg(list,OP_ADD,OS_ADDR,tmpref.base,tmpref.index,r);            if tmpref.offset<>0 then              a_op_const_reg_reg(list,OP_ADD,OS_ADDR,tmpref.offset,r,r);          end        else          begin            if tmpref.base=NR_NO then              a_load_const_reg(list,OS_ADDR,tmpref.offset,r)            else              if tmpref.offset<>0 then                a_op_const_reg_reg(list,OP_ADD,OS_ADDR,tmpref.offset,tmpref.base,r)              else                begin                  instr:=taicpu.op_reg_reg(A_MOV,r,tmpref.base);                  list.concat(instr);                  add_move_instruction(instr);                end;          end;      end;    procedure tbasecgarm.fixref(list : TAsmList;var ref : treference);      var        tmpreg, tmpreg2 : tregister;        tmpref : treference;        l, piclabel : tasmlabel;        indirection_done : boolean;      begin        { absolute symbols can't be handled directly, we've to store the symbol reference          in the text segment and access it pc relative          For now, we assume that references where base or index equals to PC are already          relative, all other references are assumed to be absolute and thus they need          to be handled extra.          A proper solution would be to change refoptions to a set and store the information          if the symbol is absolute or relative there.        }        { create consts entry }        reference_reset(tmpref,4);        current_asmdata.getjumplabel(l);        cg.a_label(current_procinfo.aktlocaldata,l);        tmpref.symboldata:=current_procinfo.aktlocaldata.last;        piclabel:=nil;        tmpreg:=NR_NO;        indirection_done:=false;        if assigned(ref.symbol) then          begin            if (target_info.system=system_arm_darwin) and               (ref.symbol.bind in [AB_EXTERNAL,AB_WEAK_EXTERNAL,AB_PRIVATE_EXTERN,AB_COMMON]) then              begin                tmpreg:=g_indirect_sym_load(list,ref.symbol.name,asmsym2indsymflags(ref.symbol));                if ref.offset<>0 then                  a_op_const_reg(list,OP_ADD,OS_ADDR,ref.offset,tmpreg);                indirection_done:=true;              end            else if (cs_create_pic in current_settings.moduleswitches) then              if (tf_pic_uses_got in target_info.flags) then                current_procinfo.aktlocaldata.concat(tai_const.Create_type_sym_offset(aitconst_got,ref.symbol,ref.offset))              else                begin                  { ideally, we would want to generate                      ldr       r1, LPICConstPool                    LPICLocal:                      ldr/str   r2,[pc,r1]                    ...                      LPICConstPool:                        .long _globsym-(LPICLocal+8)                    However, we cannot be sure that the ldr/str will follow                    right after the call to fixref, so we have to load the                    complete address already in a register.                  }                  current_asmdata.getaddrlabel(piclabel);                  current_procinfo.aktlocaldata.concat(tai_const.Create_rel_sym_offset(aitconst_ptr,piclabel,ref.symbol,ref.offset-8));                end            else              current_procinfo.aktlocaldata.concat(tai_const.create_sym_offset(ref.symbol,ref.offset))          end        else          current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(ref.offset));        { load consts entry }        if not indirection_done then          begin            tmpreg:=getintregister(list,OS_INT);            tmpref.symbol:=l;            tmpref.base:=NR_PC;            list.concat(taicpu.op_reg_ref(A_LDR,tmpreg,tmpref));            if (cs_create_pic in current_settings.moduleswitches) and               (tf_pic_uses_got in target_info.flags) and               assigned(ref.symbol) then              begin                reference_reset(tmpref,4);                tmpref.base:=current_procinfo.got;                tmpref.index:=tmpreg;                list.concat(taicpu.op_reg_ref(A_LDR,tmpreg,tmpref));              end;          end;        if assigned(piclabel) then          begin            cg.a_label(list,piclabel);            tmpreg2:=getaddressregister(list);            a_op_reg_reg_reg(list,OP_ADD,OS_ADDR,tmpreg,NR_PC,tmpreg2);            tmpreg:=tmpreg2          end;        { This routine can be called with PC as base/index in case the offset          was too large to encode in a load/store. In that case, the entire          absolute expression has been re-encoded in a new constpool entry, and          we have to remove the use of PC from the original reference (the code          above made everything relative to the value loaded from the new          constpool entry) }        if is_pc(ref.base) then          ref.base:=NR_NO;        if is_pc(ref.index) then          ref.index:=NR_NO;        if (ref.base<>NR_NO) then          begin            if ref.index<>NR_NO then              begin                list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,tmpreg));                ref.base:=tmpreg;              end            else              if ref.base<>NR_PC then                begin                  ref.index:=tmpreg;                  ref.shiftimm:=0;                  ref.signindex:=1;                  ref.shiftmode:=SM_None;                end              else                ref.base:=tmpreg;          end        else          ref.base:=tmpreg;        ref.offset:=0;        ref.symbol:=nil;      end;    procedure tbasecgarm.g_concatcopy_move(list : TAsmList;const source,dest : treference;len : tcgint);      var        paraloc1,paraloc2,paraloc3 : TCGPara;        pd : tprocdef;      begin        pd:=search_system_proc('MOVE');        paraloc1.init;        paraloc2.init;        paraloc3.init;        paramanager.getintparaloc(pd,1,paraloc1);        paramanager.getintparaloc(pd,2,paraloc2);        paramanager.getintparaloc(pd,3,paraloc3);        a_load_const_cgpara(list,OS_SINT,len,paraloc3);        a_loadaddr_ref_cgpara(list,dest,paraloc2);        a_loadaddr_ref_cgpara(list,source,paraloc1);        paramanager.freecgpara(list,paraloc3);        paramanager.freecgpara(list,paraloc2);        paramanager.freecgpara(list,paraloc1);        alloccpuregisters(list,R_INTREGISTER,paramanager.get_volatile_registers_int(pocall_default));        alloccpuregisters(list,R_FPUREGISTER,paramanager.get_volatile_registers_fpu(pocall_default));        a_call_name(list,'FPC_MOVE',false);        dealloccpuregisters(list,R_FPUREGISTER,paramanager.get_volatile_registers_fpu(pocall_default));        dealloccpuregisters(list,R_INTREGISTER,paramanager.get_volatile_registers_int(pocall_default));        paraloc3.done;        paraloc2.done;        paraloc1.done;      end;    procedure tbasecgarm.g_concatcopy_internal(list : TAsmList;const source,dest : treference;len : tcgint;aligned : boolean);      const        maxtmpreg_arm = 10; {roozbeh: can be reduced to 8 or lower if might conflick with reserved ones,also +2 is used becouse of regs required for referencing}        maxtmpreg_thumb = 5;      var        srcref,dstref,usedtmpref,usedtmpref2:treference;        srcreg,destreg,countreg,r,tmpreg:tregister;        helpsize:aint;        copysize:byte;        cgsize:Tcgsize;        tmpregisters:array[1..maxtmpreg_arm] of tregister;        maxtmpreg,        tmpregi,tmpregi2:byte;      { will never be called with count<=4 }      procedure genloop(count : aword;size : byte);        const          size2opsize : array[1..4] of tcgsize = (OS_8,OS_16,OS_NO,OS_32);        var          l : tasmlabel;        begin          current_asmdata.getjumplabel(l);          if count<size then size:=1;          a_load_const_reg(list,OS_INT,count div size,countreg);          cg.a_label(list,l);          srcref.addressmode:=AM_POSTINDEXED;          dstref.addressmode:=AM_POSTINDEXED;          srcref.offset:=size;          dstref.offset:=size;          r:=getintregister(list,size2opsize[size]);          a_load_ref_reg(list,size2opsize[size],size2opsize[size],srcref,r);          a_reg_alloc(list,NR_DEFAULTFLAGS);          list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SUB,countreg,countreg,1),PF_S));          a_load_reg_ref(list,size2opsize[size],size2opsize[size],r,dstref);          a_jmp_flags(list,F_NE,l);          a_reg_dealloc(list,NR_DEFAULTFLAGS);          srcref.offset:=1;          dstref.offset:=1;          case count mod size of            1:              begin                a_load_ref_reg(list,OS_8,OS_8,srcref,r);                a_load_reg_ref(list,OS_8,OS_8,r,dstref);              end;            2:              if aligned then                begin                  a_load_ref_reg(list,OS_16,OS_16,srcref,r);                  a_load_reg_ref(list,OS_16,OS_16,r,dstref);                end              else                begin                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                end;            3:              if aligned then                begin                  srcref.offset:=2;                  dstref.offset:=2;                  a_load_ref_reg(list,OS_16,OS_16,srcref,r);                  a_load_reg_ref(list,OS_16,OS_16,r,dstref);                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                end              else                begin                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                end;          end;          { keep the registers alive }          list.concat(taicpu.op_reg_reg(A_MOV,countreg,countreg));          list.concat(taicpu.op_reg_reg(A_MOV,srcreg,srcreg));          list.concat(taicpu.op_reg_reg(A_MOV,destreg,destreg));        end;      { will never be called with count<=4 }      procedure genloop_thumb(count : aword;size : byte);        procedure refincofs(const ref : treference;const value : longint = 1);          begin            a_op_const_reg(list,OP_ADD,OS_ADDR,value,ref.base);          end;        const          size2opsize : array[1..4] of tcgsize = (OS_8,OS_16,OS_NO,OS_32);        var          l : tasmlabel;        begin          current_asmdata.getjumplabel(l);          if count<size then size:=1;          a_load_const_reg(list,OS_INT,count div size,countreg);          cg.a_label(list,l);          r:=getintregister(list,size2opsize[size]);          a_load_ref_reg(list,size2opsize[size],size2opsize[size],srcref,r);          refincofs(srcref);          a_load_reg_ref(list,size2opsize[size],size2opsize[size],r,dstref);          refincofs(dstref);          a_reg_alloc(list,NR_DEFAULTFLAGS);          list.concat(taicpu.op_reg_reg_const(A_SUB,countreg,countreg,1));          a_jmp_flags(list,F_NE,l);          a_reg_dealloc(list,NR_DEFAULTFLAGS);          case count mod size of            1:              begin                a_load_ref_reg(list,OS_8,OS_8,srcref,r);                a_load_reg_ref(list,OS_8,OS_8,r,dstref);              end;            2:              if aligned then                begin                  a_load_ref_reg(list,OS_16,OS_16,srcref,r);                  a_load_reg_ref(list,OS_16,OS_16,r,dstref);                end              else                begin                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  refincofs(srcref);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                  refincofs(dstref);                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                end;            3:              if aligned then                begin                  a_load_ref_reg(list,OS_16,OS_16,srcref,r);                  refincofs(srcref,2);                  a_load_reg_ref(list,OS_16,OS_16,r,dstref);                  refincofs(dstref,2);                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                end              else                begin                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  refincofs(srcref);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                  refincofs(dstref);                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  refincofs(srcref);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                  refincofs(dstref);                  a_load_ref_reg(list,OS_8,OS_8,srcref,r);                  a_load_reg_ref(list,OS_8,OS_8,r,dstref);                end;          end;          { keep the registers alive }          list.concat(taicpu.op_reg_reg(A_MOV,countreg,countreg));          list.concat(taicpu.op_reg_reg(A_MOV,srcreg,srcreg));          list.concat(taicpu.op_reg_reg(A_MOV,destreg,destreg));        end;      begin        if len=0 then          exit;        if GenerateThumbCode then          maxtmpreg:=maxtmpreg_thumb        else          maxtmpreg:=maxtmpreg_arm;        helpsize:=12+maxtmpreg*4;//52 with maxtmpreg=10        dstref:=dest;        srcref:=source;        if cs_opt_size in current_settings.optimizerswitches then          helpsize:=8;        if aligned and (len=4) then          begin            tmpreg:=getintregister(list,OS_32);            a_load_ref_reg(list,OS_32,OS_32,source,tmpreg);            a_load_reg_ref(list,OS_32,OS_32,tmpreg,dest);          end        else if aligned and (len=2) then          begin            tmpreg:=getintregister(list,OS_16);            a_load_ref_reg(list,OS_16,OS_16,source,tmpreg);            a_load_reg_ref(list,OS_16,OS_16,tmpreg,dest);          end        else if (len<=helpsize) and aligned then          begin            tmpregi:=0;            srcreg:=getintregister(list,OS_ADDR);            { explicit pc relative addressing, could be              e.g. a floating point constant }            if source.base=NR_PC then              begin                { ... then we don't need a loadaddr }                srcref:=source;              end            else              begin                a_loadaddr_ref_reg(list,source,srcreg);                reference_reset_base(srcref,srcreg,0,source.alignment);              end;            while (len div 4 <> 0) and (tmpregi<maxtmpreg) do              begin                inc(tmpregi);                tmpregisters[tmpregi]:=getintregister(list,OS_32);                a_load_ref_reg(list,OS_32,OS_32,srcref,tmpregisters[tmpregi]);                inc(srcref.offset,4);                dec(len,4);              end;            destreg:=getintregister(list,OS_ADDR);            a_loadaddr_ref_reg(list,dest,destreg);            reference_reset_base(dstref,destreg,0,dest.alignment);            tmpregi2:=1;            while (tmpregi2<=tmpregi) do              begin                a_load_reg_ref(list,OS_32,OS_32,tmpregisters[tmpregi2],dstref);                inc(dstref.offset,4);                inc(tmpregi2);              end;            copysize:=4;            cgsize:=OS_32;            while len<>0 do              begin                if len<2 then                  begin                    copysize:=1;                    cgsize:=OS_8;                  end                else if len<4 then                  begin                    copysize:=2;                    cgsize:=OS_16;                  end;                dec(len,copysize);                r:=getintregister(list,cgsize);                a_load_ref_reg(list,cgsize,cgsize,srcref,r);                a_load_reg_ref(list,cgsize,cgsize,r,dstref);                inc(srcref.offset,copysize);                inc(dstref.offset,copysize);              end;{end of while}          end        else          begin            cgsize:=OS_32;            if (len<=4) then{len<=4 and not aligned}              begin                r:=getintregister(list,cgsize);                usedtmpref:=a_internal_load_ref_reg(list,OS_8,OS_8,srcref,r);                if Len=1 then                  a_load_reg_ref(list,OS_8,OS_8,r,dstref)                else                  begin                    tmpreg:=getintregister(list,cgsize);                    usedtmpref2:=a_internal_load_reg_ref(list,OS_8,OS_8,r,dstref);                    inc(usedtmpref.offset,1);                    a_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                    inc(usedtmpref2.offset,1);                    a_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref2);                    if len>2 then                      begin                        inc(usedtmpref.offset,1);                        a_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                        inc(usedtmpref2.offset,1);                        a_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref2);                        if len>3 then                          begin                            inc(usedtmpref.offset,1);                            a_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                            inc(usedtmpref2.offset,1);                            a_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref2);                          end;                        end;                      end;            end{end of if len<=4}            else              begin{unaligned & 4<len<helpsize **or** aligned/unaligned & len>helpsize}                destreg:=getintregister(list,OS_ADDR);                a_loadaddr_ref_reg(list,dest,destreg);                reference_reset_base(dstref,destreg,0,dest.alignment);                srcreg:=getintregister(list,OS_ADDR);                a_loadaddr_ref_reg(list,source,srcreg);                reference_reset_base(srcref,srcreg,0,source.alignment);                countreg:=getintregister(list,OS_32);//            if cs_opt_size in current_settings.optimizerswitches  then                { roozbeh : it seems loading 1 byte is faster becouse of caching/fetching(?) }                {if aligned then                genloop(len,4)                else}                if GenerateThumbCode then                  genloop_thumb(len,1)                else                  genloop(len,1);            end;          end;    end;    procedure tbasecgarm.g_concatcopy_unaligned(list : TAsmList;const source,dest : treference;len : tcgint);      begin        g_concatcopy_internal(list,source,dest,len,false);      end;    procedure tbasecgarm.g_concatcopy(list : TAsmList;const source,dest : treference;len : tcgint);      begin        if (source.alignment in [1,3]) or          (dest.alignment in [1,3]) then          g_concatcopy_internal(list,source,dest,len,false)        else          g_concatcopy_internal(list,source,dest,len,true);      end;    procedure tbasecgarm.g_overflowCheck(list : TAsmList;const l : tlocation;def : tdef);      var        ovloc : tlocation;      begin        ovloc.loc:=LOC_VOID;        g_overflowCheck_loc(list,l,def,ovloc);      end;    procedure tbasecgarm.g_overflowCheck_loc(List:TAsmList;const Loc:TLocation;def:TDef;ovloc : tlocation);      var        hl : tasmlabel;        ai:TAiCpu;        hflags : tresflags;      begin        if not(cs_check_overflow in current_settings.localswitches) then          exit;        current_asmdata.getjumplabel(hl);        case ovloc.loc of          LOC_VOID:            begin              ai:=taicpu.op_sym(A_B,hl);              ai.is_jmp:=true;              if not((def.typ=pointerdef) or                    ((def.typ=orddef) and                     (torddef(def).ordtype in [u64bit,u16bit,u32bit,u8bit,uchar,                                               pasbool8,pasbool16,pasbool32,pasbool64]))) then                 ai.SetCondition(C_VC)              else                if TAiCpu(List.Last).opcode in [A_RSB,A_RSC,A_SBC,A_SUB] then                  ai.SetCondition(C_CS)                else                  ai.SetCondition(C_CC);              list.concat(ai);            end;          LOC_FLAGS:            begin              hflags:=ovloc.resflags;              inverse_flags(hflags);              cg.a_jmp_flags(list,hflags,hl);              cg.a_reg_dealloc(list,NR_DEFAULTFLAGS);            end;          else            internalerror(200409281);        end;        a_call_name(list,'FPC_OVERFLOW',false);        a_label(list,hl);      end;    procedure tbasecgarm.g_save_registers(list : TAsmList);      begin        { this work is done in g_proc_entry }      end;    procedure tbasecgarm.g_restore_registers(list : TAsmList);      begin        { this work is done in g_proc_exit }      end;    procedure tbasecgarm.a_jmp_cond(list : TAsmList;cond : TOpCmp;l: tasmlabel);      var        ai : taicpu;        hlabel : TAsmLabel;      begin        if GenerateThumbCode then          begin            { the optimizer has to fix this if jump range is sufficient short }            current_asmdata.getjumplabel(hlabel);            ai:=Taicpu.Op_sym(A_B,hlabel);            ai.SetCondition(inverse_cond(OpCmp2AsmCond[cond]));            ai.is_jmp:=true;            list.concat(ai);            a_jmp_always(list,l);            a_label(list,hlabel);          end        else          begin            ai:=Taicpu.Op_sym(A_B,l);            ai.SetCondition(OpCmp2AsmCond[cond]);            ai.is_jmp:=true;            list.concat(ai);          end;      end;    function get_scalar_mm_op(fromsize,tosize : tcgsize) : tasmop;      const        convertop : array[OS_F32..OS_F128,OS_F32..OS_F128] of tasmop = (          (A_FCPYS,A_FCVTSD,A_NONE,A_NONE,A_NONE),          (A_FCVTDS,A_FCPYD,A_NONE,A_NONE,A_NONE),          (A_NONE,A_NONE,A_NONE,A_NONE,A_NONE),          (A_NONE,A_NONE,A_NONE,A_NONE,A_NONE),          (A_NONE,A_NONE,A_NONE,A_NONE,A_NONE));      begin        result:=convertop[fromsize,tosize];        if result=A_NONE then          internalerror(200312205);      end;    procedure tbasecgarm.a_loadmm_reg_reg(list: tasmlist; fromsize,tosize: tcgsize; reg1,reg2: tregister; shuffle: pmmshuffle);      var        instr: taicpu;      begin        if shuffle=nil then          begin            if fromsize=tosize then              { needs correct size in case of spilling }              case fromsize of                OS_F32:                  instr:=taicpu.op_reg_reg(A_FCPYS,reg2,reg1);                OS_F64:                  instr:=taicpu.op_reg_reg(A_FCPYD,reg2,reg1);                else                  internalerror(2009112405);              end            else              internalerror(2009112406);          end        else if shufflescalar(shuffle) then          instr:=taicpu.op_reg_reg(get_scalar_mm_op(tosize,fromsize),reg2,reg1)        else          internalerror(2009112407);        list.concat(instr);        case instr.opcode of          A_FCPYS,          A_FCPYD:            add_move_instruction(instr);        end;      end;    procedure tbasecgarm.a_loadmm_ref_reg(list: tasmlist; fromsize,tosize: tcgsize; const ref: treference; reg: tregister; shuffle: pmmshuffle);      var        intreg,        tmpmmreg : tregister;        reg64    : tregister64;        op       : tasmop;      begin        if assigned(shuffle) and           not(shufflescalar(shuffle)) then          internalerror(2009112413);        case fromsize of          OS_32,OS_S32:            begin              fromsize:=OS_F32;              { since we are loading an integer, no conversion may be required }              if (fromsize<>tosize) then                internalerror(2009112801);            end;          OS_64,OS_S64:            begin              fromsize:=OS_F64;              { since we are loading an integer, no conversion may be required }              if (fromsize<>tosize) then                internalerror(2009112901);            end;        end;        if (fromsize<>tosize) then          tmpmmreg:=getmmregister(list,fromsize)        else          tmpmmreg:=reg;        if (ref.alignment in [1,2]) then          begin            case fromsize of              OS_F32:                begin                  intreg:=getintregister(list,OS_32);                  a_load_ref_reg(list,OS_32,OS_32,ref,intreg);                  a_loadmm_intreg_reg(list,OS_32,OS_F32,intreg,tmpmmreg,mms_movescalar);                end;              OS_F64:                begin                  reg64.reglo:=getintregister(list,OS_32);                  reg64.reghi:=getintregister(list,OS_32);                  cg64.a_load64_ref_reg(list,ref,reg64);                  cg64.a_loadmm_intreg64_reg(list,OS_F64,reg64,tmpmmreg);                end;              else                internalerror(2009112412);            end;          end        else          begin             case fromsize of               OS_F32:                 op:=A_FLDS;               OS_F64:                 op:=A_FLDD;               else                 internalerror(2009112415);             end;             handle_load_store(list,op,PF_None,tmpmmreg,ref);          end;        if (tmpmmreg<>reg) then          a_loadmm_reg_reg(list,fromsize,tosize,tmpmmreg,reg,shuffle);      end;    procedure tbasecgarm.a_loadmm_reg_ref(list: tasmlist; fromsize,tosize: tcgsize; reg: tregister; const ref: treference; shuffle: pmmshuffle);      var        intreg,        tmpmmreg : tregister;        reg64    : tregister64;        op       : tasmop;      begin        if assigned(shuffle) and           not(shufflescalar(shuffle)) then          internalerror(2009112416);        case tosize of          OS_32,OS_S32:            begin              tosize:=OS_F32;              { since we are loading an integer, no conversion may be required }              if (fromsize<>tosize) then                internalerror(2009112801);            end;          OS_64,OS_S64:            begin              tosize:=OS_F64;              { since we are loading an integer, no conversion may be required }              if (fromsize<>tosize) then                internalerror(2009112901);            end;        end;        if (fromsize<>tosize) then          begin            tmpmmreg:=getmmregister(list,tosize);            a_loadmm_reg_reg(list,fromsize,tosize,reg,tmpmmreg,shuffle);          end        else          tmpmmreg:=reg;        if (ref.alignment in [1,2]) then          begin            case tosize of              OS_F32:                begin                  intreg:=getintregister(list,OS_32);                  a_loadmm_reg_intreg(list,OS_F32,OS_32,tmpmmreg,intreg,shuffle);                  a_load_reg_ref(list,OS_32,OS_32,intreg,ref);                end;              OS_F64:                begin                  reg64.reglo:=getintregister(list,OS_32);                  reg64.reghi:=getintregister(list,OS_32);                  cg64.a_loadmm_reg_intreg64(list,OS_F64,tmpmmreg,reg64);                  cg64.a_load64_reg_ref(list,reg64,ref);                end;              else                internalerror(2009112417);            end;          end        else          begin             case fromsize of               OS_F32:                 op:=A_FSTS;               OS_F64:                 op:=A_FSTD;               else                 internalerror(2009112418);             end;             handle_load_store(list,op,PF_None,tmpmmreg,ref);          end;      end;    procedure tbasecgarm.a_loadmm_intreg_reg(list: TAsmList; fromsize, tosize : tcgsize; intreg, mmreg: tregister; shuffle: pmmshuffle);      begin        { this code can only be used to transfer raw data, not to perform          conversions }        if (tosize<>OS_F32) then          internalerror(2009112419);        if not(fromsize in [OS_32,OS_S32]) then          internalerror(2009112420);        if assigned(shuffle) and           not shufflescalar(shuffle) then          internalerror(2009112516);        list.concat(taicpu.op_reg_reg(A_FMSR,mmreg,intreg));      end;    procedure tbasecgarm.a_loadmm_reg_intreg(list: TAsmList; fromsize, tosize : tcgsize; mmreg, intreg: tregister;shuffle : pmmshuffle);      begin        { this code can only be used to transfer raw data, not to perform          conversions }        if (fromsize<>OS_F32) then          internalerror(2009112430);        if not(tosize in [OS_32,OS_S32]) then          internalerror(2009112420);        if assigned(shuffle) and           not shufflescalar(shuffle) then          internalerror(2009112514);        list.concat(taicpu.op_reg_reg(A_FMRS,intreg,mmreg));      end;      procedure tbasecgarm.a_opmm_reg_reg(list: tasmlist; op: topcg; size: tcgsize; src, dst: tregister; shuffle: pmmshuffle);        var          tmpreg: tregister;        begin          { the vfp doesn't support xor nor any other logical operation, but            this routine is used to initialise global mm regvars. We can            easily initialise an mm reg with 0 though. }          case op of            OP_XOR:              begin                if (src<>dst) or                   (reg_cgsize(src)<>size) or                   assigned(shuffle) then                  internalerror(2009112907);                tmpreg:=getintregister(list,OS_32);                a_load_const_reg(list,OS_32,0,tmpreg);                case size of                  OS_F32:                    list.concat(taicpu.op_reg_reg(A_FMSR,dst,tmpreg));                  OS_F64:                    list.concat(taicpu.op_reg_reg_reg(A_FMDRR,dst,tmpreg,tmpreg));                  else                    internalerror(2009112908);                end;              end            else              internalerror(2009112906);          end;        end;    procedure tbasecgarm.g_intf_wrapper(list: TAsmList; procdef: tprocdef; const labelname: string; ioffset: longint);      procedure loadvmttor12;        var          tmpref,          href : treference;          extrareg : boolean;          l : TAsmLabel;        begin          reference_reset_base(href,NR_R0,0,sizeof(pint));          if GenerateThumbCode then            begin              if (href.offset in [0..124]) and ((href.offset mod 4)=0) then                begin                  list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,[RS_R0]));                  cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R0);                  list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_R0));                  list.concat(taicpu.op_regset(A_POP,R_INTREGISTER,R_SUBWHOLE,[RS_R0]));                end              else                begin                  list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,[RS_R0,RS_R1]));                  { create consts entry }                  reference_reset(tmpref,4);                  current_asmdata.getjumplabel(l);                  current_procinfo.aktlocaldata.Concat(tai_align.Create(4));                  cg.a_label(current_procinfo.aktlocaldata,l);                  tmpref.symboldata:=current_procinfo.aktlocaldata.last;                  current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(href.offset));                  tmpref.symbol:=l;                  tmpref.base:=NR_PC;                  list.concat(taicpu.op_reg_ref(A_LDR,NR_R1,tmpref));                  href.offset:=0;                  href.index:=NR_R1;                  cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R0);                  list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_R0));                  list.concat(taicpu.op_regset(A_POP,R_INTREGISTER,R_SUBWHOLE,[RS_R0,RS_R1]));                end;            end          else            cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R12);        end;      procedure op_onr12methodaddr;        var          tmpref,          href : treference;          l : TAsmLabel;        begin          if (procdef.extnumber=$ffff) then            Internalerror(200006139);          if GenerateThumbCode then            begin              reference_reset_base(href,NR_R0,tobjectdef(procdef.struct).vmtmethodoffset(procdef.extnumber),sizeof(pint));              if (href.offset in [0..124]) and ((href.offset mod 4)=0) then                begin                  list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,[RS_R0]));                  list.concat(taicpu.op_reg_reg(A_MOV,NR_R0,NR_R12));                  cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R0);                  list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_R0));                  list.concat(taicpu.op_regset(A_POP,R_INTREGISTER,R_SUBWHOLE,[RS_R0]));                end              else                begin                  list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,[RS_R0,RS_R1]));                  { create consts entry }                  reference_reset(tmpref,4);                  current_asmdata.getjumplabel(l);                  current_procinfo.aktlocaldata.Concat(tai_align.Create(4));                  cg.a_label(current_procinfo.aktlocaldata,l);                  tmpref.symboldata:=current_procinfo.aktlocaldata.last;                  current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(href.offset));                  tmpref.symbol:=l;                  tmpref.base:=NR_PC;                  list.concat(taicpu.op_reg_ref(A_LDR,NR_R1,tmpref));                  list.concat(taicpu.op_reg_reg(A_MOV,NR_R0,NR_R12));                  href.offset:=0;                  href.base:=NR_R0;                  href.index:=NR_R1;                  cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R0);                  list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_R0));                  list.concat(taicpu.op_regset(A_POP,R_INTREGISTER,R_SUBWHOLE,[RS_R0,RS_R1]));                end;            end          else            begin              reference_reset_base(href,NR_R12,tobjectdef(procdef.struct).vmtmethodoffset(procdef.extnumber),sizeof(pint));              cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R12);            end;          if not(CPUARM_HAS_BX in cpu_capabilities[current_settings.cputype]) then            list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R12))          else            list.concat(taicpu.op_reg(A_BX,NR_R12));        end;      var        make_global : boolean;        tmpref : treference;        l : TAsmLabel;      begin        if not(procdef.proctypeoption in [potype_function,potype_procedure]) then          Internalerror(200006137);        if not assigned(procdef.struct) or           (procdef.procoptions*[po_classmethod, po_staticmethod,             po_methodpointer, po_interrupt, po_iocheck]<>[]) then          Internalerror(200006138);        if procdef.owner.symtabletype<>ObjectSymtable then          Internalerror(200109191);          if GenerateThumbCode or GenerateThumb2Code then            list.concat(tai_thumb_func.create);        make_global:=false;        if (not current_module.is_unit) or           create_smartlink or           (procdef.owner.defowner.owner.symtabletype=globalsymtable) then          make_global:=true;        if make_global then          list.concat(Tai_symbol.Createname_global(labelname,AT_FUNCTION,0))        else          list.concat(Tai_symbol.Createname(labelname,AT_FUNCTION,0));        { the wrapper might need aktlocaldata for the additional data to          load the constant }        current_procinfo:=cprocinfo.create(nil);        { set param1 interface to self  }        g_adjust_self_value(list,procdef,ioffset);        { case 4 }        if (po_virtualmethod in procdef.procoptions) and            not is_objectpascal_helper(procdef.struct) then          begin            loadvmttor12;            op_onr12methodaddr;          end        { case 0 }        else if GenerateThumbCode then          begin            { bl cannot be used here because it destroys lr }            list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,[RS_R0]));            { create consts entry }            reference_reset(tmpref,4);            current_asmdata.getjumplabel(l);            current_procinfo.aktlocaldata.Concat(tai_align.Create(4));            cg.a_label(current_procinfo.aktlocaldata,l);            tmpref.symboldata:=current_procinfo.aktlocaldata.last;            current_procinfo.aktlocaldata.concat(tai_const.Create_sym(current_asmdata.RefAsmSymbol(procdef.mangledname)));            tmpref.symbol:=l;            tmpref.base:=NR_PC;            cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,tmpref,NR_R0);            list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_R0));            list.concat(taicpu.op_regset(A_POP,R_INTREGISTER,R_SUBWHOLE,[RS_R0]));            list.concat(taicpu.op_reg(A_BX,NR_R12));          end        else          list.concat(taicpu.op_sym(A_B,current_asmdata.RefAsmSymbol(procdef.mangledname)));        list.concatlist(current_procinfo.aktlocaldata);        current_procinfo.Free;        current_procinfo:=nil;        list.concat(Tai_symbol_end.Createname(labelname));      end;    procedure tbasecgarm.maybeadjustresult(list: TAsmList; op: TOpCg; size: tcgsize; dst: tregister);      const        overflowops = [OP_MUL,OP_SHL,OP_ADD,OP_SUB,OP_NEG];      begin        if (op in overflowops) and           (size in [OS_8,OS_S8,OS_16,OS_S16]) then          a_load_reg_reg(list,OS_32,size,dst,dst);      end;    procedure tbasecgarm.safe_mla(list : TAsmList; op1,op2,op3,op4 : TRegister);      procedure checkreg(var reg : TRegister);        var          tmpreg : TRegister;        begin          if ((GenerateThumbCode or GenerateThumb2Code) and (getsupreg(reg)=RS_R13)) or            (getsupreg(reg)=RS_R15) then            begin              tmpreg:=getintregister(list,OS_INT);              a_load_reg_reg(list,OS_INT,OS_INT,reg,tmpreg);              reg:=tmpreg;            end;        end;      begin        checkreg(op1);        checkreg(op2);        checkreg(op3);        checkreg(op4);        list.concat(taicpu.op_reg_reg_reg_reg(A_MLA,op1,op2,op3,op4));      end;    procedure tcg64farm.a_op64_reg_reg(list : TAsmList;op:TOpCG;size : tcgsize;regsrc,regdst : tregister64);      begin        case op of          OP_NEG:            begin              cg.a_reg_alloc(list,NR_DEFAULTFLAGS);              list.concat(setoppostfix(taicpu.op_reg_reg_const(A_RSB,regdst.reglo,regsrc.reglo,0),PF_S));              list.concat(taicpu.op_reg_reg_const(A_RSC,regdst.reghi,regsrc.reghi,0));              cg.a_reg_dealloc(list,NR_DEFAULTFLAGS);            end;          OP_NOT:            begin              cg.a_op_reg_reg(list,OP_NOT,OS_INT,regsrc.reglo,regdst.reglo);              cg.a_op_reg_reg(list,OP_NOT,OS_INT,regsrc.reghi,regdst.reghi);            end;          else            a_op64_reg_reg_reg(list,op,size,regsrc,regdst,regdst);        end;      end;    procedure tcg64farm.a_op64_const_reg(list : TAsmList;op:TOpCG;size : tcgsize;value : int64;reg : tregister64);      begin        a_op64_const_reg_reg(list,op,size,value,reg,reg);      end;    procedure tcg64farm.a_op64_const_reg_reg(list: TAsmList;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64);      var        ovloc : tlocation;      begin        a_op64_const_reg_reg_checkoverflow(list,op,size,value,regsrc,regdst,false,ovloc);      end;    procedure tcg64farm.a_op64_reg_reg_reg(list: TAsmList;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64);      var        ovloc : tlocation;      begin        a_op64_reg_reg_reg_checkoverflow(list,op,size,regsrc1,regsrc2,regdst,false,ovloc);      end;    procedure tcg64farm.a_loadmm_intreg64_reg(list: TAsmList; mmsize: tcgsize; intreg: tregister64; mmreg: tregister);      begin        { this code can only be used to transfer raw data, not to perform          conversions }        if (mmsize<>OS_F64) then          internalerror(2009112405);        list.concat(taicpu.op_reg_reg_reg(A_FMDRR,mmreg,intreg.reglo,intreg.reghi));      end;    procedure tcg64farm.a_loadmm_reg_intreg64(list: TAsmList; mmsize: tcgsize; mmreg: tregister; intreg: tregister64);      begin        { this code can only be used to transfer raw data, not to perform          conversions }        if (mmsize<>OS_F64) then          internalerror(2009112406);        list.concat(taicpu.op_reg_reg_reg(A_FMRRD,intreg.reglo,intreg.reghi,mmreg));      end;    procedure tcg64farm.a_op64_const_reg_reg_checkoverflow(list: TAsmList;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64;setflags : boolean;var ovloc : tlocation);      var        tmpreg : tregister;        b : byte;      begin        ovloc.loc:=LOC_VOID;        case op of          OP_NEG,          OP_NOT :            internalerror(2012022501);        end;        if (setflags or tbasecgarm(cg).cgsetflags) and (op in [OP_ADD,OP_SUB]) then          begin            case op of              OP_ADD:                begin                  if is_shifter_const(lo(value),b) then                    begin                      cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                      list.concat(setoppostfix(taicpu.op_reg_reg_const(A_ADD,regdst.reglo,regsrc.reglo,lo(value)),PF_S))                    end                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);                      cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc.reglo,tmpreg),PF_S));                    end;                  if is_shifter_const(hi(value),b) then                    list.concat(setoppostfix(taicpu.op_reg_reg_const(A_ADC,regdst.reghi,regsrc.reghi,hi(value)),PF_S))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc.reghi,tmpreg),PF_S));                    end;                end;              OP_SUB:                begin                  if is_shifter_const(lo(value),b) then                    begin                      cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                      list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SUB,regdst.reglo,regsrc.reglo,lo(value)),PF_S))                    end                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);                      cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc.reglo,tmpreg),PF_S));                    end;                  if is_shifter_const(hi(value),b) then                    list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SBC,regdst.reghi,regsrc.reghi,aint(hi(value))),PF_S))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc.reghi,tmpreg),PF_S));                    end;                end;              else                internalerror(200502131);            end;            if size=OS_64 then              begin                { the arm has an weired opinion how flags for SUB/ADD are handled }                ovloc.loc:=LOC_FLAGS;                case op of                  OP_ADD:                    ovloc.resflags:=F_CS;                  OP_SUB:                    ovloc.resflags:=F_CC;                end;              end;          end        else          begin            case op of              OP_AND,OP_OR,OP_XOR:                begin                  cg.a_op_const_reg_reg(list,op,OS_32,aint(lo(value)),regsrc.reglo,regdst.reglo);                  cg.a_op_const_reg_reg(list,op,OS_32,aint(hi(value)),regsrc.reghi,regdst.reghi);                end;              OP_ADD:                begin                  if is_shifter_const(aint(lo(value)),b) then                    begin                      cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                      list.concat(setoppostfix(taicpu.op_reg_reg_const(A_ADD,regdst.reglo,regsrc.reglo,aint(lo(value))),PF_S))                    end                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,aint(lo(value)),tmpreg);                      cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc.reglo,tmpreg),PF_S));                    end;                  if is_shifter_const(aint(hi(value)),b) then                    list.concat(taicpu.op_reg_reg_const(A_ADC,regdst.reghi,regsrc.reghi,aint(hi(value))))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,aint(hi(value)),tmpreg);                      list.concat(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc.reghi,tmpreg));                    end;                end;              OP_SUB:                begin                  if is_shifter_const(aint(lo(value)),b) then                    begin                      cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                      list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SUB,regdst.reglo,regsrc.reglo,aint(lo(value))),PF_S))                    end                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,aint(lo(value)),tmpreg);                      cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                      list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc.reglo,tmpreg),PF_S));                    end;                  if is_shifter_const(aint(hi(value)),b) then                    list.concat(taicpu.op_reg_reg_const(A_SBC,regdst.reghi,regsrc.reghi,aint(hi(value))))                  else                    begin                      tmpreg:=cg.getintregister(list,OS_32);                      cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);                      list.concat(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc.reghi,tmpreg));                    end;                end;            else              internalerror(2003083101);          end;        end;      end;    procedure tcg64farm.a_op64_reg_reg_reg_checkoverflow(list: TAsmList;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64;setflags : boolean;var ovloc : tlocation);      begin        ovloc.loc:=LOC_VOID;        case op of          OP_NEG,          OP_NOT :            internalerror(2012022502);        end;        if (setflags or tbasecgarm(cg).cgsetflags) and (op in [OP_ADD,OP_SUB]) then          begin            case op of              OP_ADD:                begin                  cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc1.reglo,regsrc2.reglo),PF_S));                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc1.reghi,regsrc2.reghi),PF_S));                end;              OP_SUB:                begin                  cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc2.reglo,regsrc1.reglo),PF_S));                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc2.reghi,regsrc1.reghi),PF_S));                end;              else                internalerror(2003083101);            end;            if size=OS_64 then              begin                { the arm has an weired opinion how flags for SUB/ADD are handled }                ovloc.loc:=LOC_FLAGS;                case op of                  OP_ADD:                    ovloc.resflags:=F_CS;                  OP_SUB:                    ovloc.resflags:=F_CC;                end;              end;          end        else          begin            case op of              OP_AND,OP_OR,OP_XOR:                begin                  cg.a_op_reg_reg_reg(list,op,OS_32,regsrc1.reglo,regsrc2.reglo,regdst.reglo);                  cg.a_op_reg_reg_reg(list,op,OS_32,regsrc1.reghi,regsrc2.reghi,regdst.reghi);                end;              OP_ADD:                begin                  cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc1.reglo,regsrc2.reglo),PF_S));                  list.concat(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc1.reghi,regsrc2.reghi));                  cg.a_reg_dealloc(list,NR_DEFAULTFLAGS);                end;              OP_SUB:                begin                  cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc2.reglo,regsrc1.reglo),PF_S));                  list.concat(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc2.reghi,regsrc1.reghi));                  cg.a_reg_dealloc(list,NR_DEFAULTFLAGS);                end;              else                internalerror(2003083101);            end;          end;      end;    procedure tthumbcgarm.init_register_allocators;      begin        inherited init_register_allocators;        if assigned(current_procinfo) and (current_procinfo.framepointer=NR_R7) then          rg[R_INTREGISTER]:=trgintcputhumb.create(R_INTREGISTER,R_SUBWHOLE,              [RS_R0,RS_R1,RS_R2,RS_R3,RS_R4,RS_R5,RS_R6],first_int_imreg,[])        else          rg[R_INTREGISTER]:=trgintcputhumb.create(R_INTREGISTER,R_SUBWHOLE,              [RS_R0,RS_R1,RS_R2,RS_R3,RS_R4,RS_R5,RS_R6,RS_R7],first_int_imreg,[]);      end;    procedure tthumbcgarm.done_register_allocators;      begin        rg[R_INTREGISTER].free;        rg[R_FPUREGISTER].free;        rg[R_MMREGISTER].free;        inherited done_register_allocators;      end;    procedure tthumbcgarm.g_proc_entry(list : TAsmList;localsize : longint;nostackframe:boolean);      var         ref : treference;         shift : byte;         r : byte;         regs, saveregs : tcpuregisterset;         r7offset,         stackmisalignment : pint;         postfix: toppostfix;         registerarea,         imm1, imm2: DWord;         stack_parameters: Boolean;      begin        stack_parameters:=current_procinfo.procdef.stack_tainting_parameter(calleeside);        LocalSize:=align(LocalSize,4);        { call instruction does not put anything on the stack }        stackmisalignment:=0;        if not(nostackframe) then          begin            a_reg_alloc(list,NR_STACK_POINTER_REG);            if current_procinfo.framepointer<>NR_STACK_POINTER_REG then              a_reg_alloc(list,NR_FRAME_POINTER_REG);            { save int registers }            reference_reset(ref,4);            ref.index:=NR_STACK_POINTER_REG;            ref.addressmode:=AM_PREINDEXED;            regs:=rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall);            if current_procinfo.framepointer<>NR_STACK_POINTER_REG then              begin                //!!!! a_reg_alloc(list,NR_R12);                //!!!! list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_STACK_POINTER_REG));              end;            { the (old) ARM APCS requires saving both the stack pointer (to              crawl the stack) and the PC (to identify the function this              stack frame belongs to) -> also save R12 (= copy of R13 on entry)              and R15 -- still needs updating for EABI and Darwin, they don't              need that }            if current_procinfo.framepointer<>NR_STACK_POINTER_REG then              regs:=regs+[RS_R7,RS_R14]            else              // if (regs<>[]) or (pi_do_call in current_procinfo.flags) then              include(regs,RS_R14);            { safely estimate stack size }            if localsize+current_settings.alignment.localalignmax+4>508 then              begin                include(rg[R_INTREGISTER].used_in_proc,RS_R4);                include(regs,RS_R4);              end;            registerarea:=0;            if regs<>[] then               begin                 for r:=RS_R0 to RS_R15 do                   if r in regs then                     inc(registerarea,4);                 list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,regs));               end;            stackmisalignment:=registerarea mod current_settings.alignment.localalignmax;            if stack_parameters or (LocalSize<>0) or               ((stackmisalignment<>0) and                ((pi_do_call in current_procinfo.flags) or                 (po_assembler in current_procinfo.procdef.procoptions))) then              begin                { do we access stack parameters?                  if yes, the previously estimated stacksize must be used }                if stack_parameters then                  begin                    if localsize>tarmprocinfo(current_procinfo).stackframesize then                      begin                        writeln(localsize);                        writeln(tarmprocinfo(current_procinfo).stackframesize);                        internalerror(2013040601);                      end                    else                      localsize:=tarmprocinfo(current_procinfo).stackframesize-registerarea;                  end                else                  localsize:=align(localsize+stackmisalignment,current_settings.alignment.localalignmax)-stackmisalignment;                if localsize<508 then                  begin                    list.concat(taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,LocalSize));                  end                else if localsize<=1016 then                  begin                    list.concat(taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,508));                    list.concat(taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,LocalSize-508));                  end                else                  begin                    a_load_const_reg(list,OS_ADDR,-localsize,NR_R4);                    list.concat(taicpu.op_reg_reg_reg(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R4));                    include(regs,RS_R4);                    //!!!! if current_procinfo.framepointer=NR_STACK_POINTER_REG then                    //!!!!   a_reg_alloc(list,NR_R12);                    //!!!! a_load_const_reg(list,OS_ADDR,LocalSize,NR_R12);                    //!!!! list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R12));                    //!!!! a_reg_dealloc(list,NR_R12);                  end;              end;            if current_procinfo.framepointer<>NR_STACK_POINTER_REG then              begin                list.concat(taicpu.op_reg_reg_const(A_ADD,current_procinfo.framepointer,NR_STACK_POINTER_REG,0));              end;          end;      end;    procedure tthumbcgarm.g_proc_exit(list: TAsmList; parasize: longint; nostackframe: boolean);      var         ref : treference;         LocalSize : longint;         r,         shift : byte;         saveregs,         regs : tcpuregisterset;         registerarea : DWord;         stackmisalignment: pint;         imm1, imm2: DWord;         stack_parameters : Boolean;      begin        if not(nostackframe) then          begin            stack_parameters:=current_procinfo.procdef.stack_tainting_parameter(calleeside);            regs:=rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall);            include(regs,RS_R15);            if current_procinfo.framepointer<>NR_STACK_POINTER_REG then              include(regs,getsupreg(current_procinfo.framepointer));            registerarea:=0;            for r:=RS_R0 to RS_R15 do              if r in regs then                inc(registerarea,4);            stackmisalignment:=registerarea mod current_settings.alignment.localalignmax;            LocalSize:=current_procinfo.calc_stackframe_size;            if stack_parameters then              localsize:=tarmprocinfo(current_procinfo).stackframesize-registerarea            else              localsize:=align(localsize+stackmisalignment,current_settings.alignment.localalignmax)-stackmisalignment;            if (current_procinfo.framepointer=NR_STACK_POINTER_REG) or               (target_info.system in systems_darwin) then              begin                if (LocalSize<>0) or                   ((stackmisalignment<>0) and                    ((pi_do_call in current_procinfo.flags) or                     (po_assembler in current_procinfo.procdef.procoptions))) then                  begin                    if LocalSize=0 then                    else if LocalSize<=508 then                      list.concat(taicpu.op_reg_reg_const(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,LocalSize))                    else if LocalSize<=1016 then                      begin                        list.concat(taicpu.op_reg_reg_const(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,508));                        list.concat(taicpu.op_reg_reg_const(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,localsize-508));                      end                    else                      begin                        a_reg_alloc(list,NR_R3);                        a_load_const_reg(list,OS_ADDR,LocalSize,NR_R3);                        list.concat(taicpu.op_reg_reg_reg(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R3));                        a_reg_dealloc(list,NR_R3);                      end;                  end;                if regs=[] then                  begin                    if not(CPUARM_HAS_BX in cpu_capabilities[current_settings.cputype]) then                      list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R14))                    else                      list.concat(taicpu.op_reg(A_BX,NR_R14))                  end                else                  list.concat(taicpu.op_regset(A_POP,R_INTREGISTER,R_SUBWHOLE,regs));              end;          end        else if not(CPUARM_HAS_BX in cpu_capabilities[current_settings.cputype]) then          list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R14))        else          list.concat(taicpu.op_reg(A_BX,NR_R14))      end;     procedure tthumbcgarm.a_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister);       var         oppostfix:toppostfix;         usedtmpref: treference;         tmpreg,tmpreg2 : tregister;         dir : integer;       begin         if (TCGSize2Size[FromSize] >= TCGSize2Size[ToSize]) then           FromSize := ToSize;         case FromSize of           { signed integer registers }           OS_8:             oppostfix:=PF_B;           OS_S8:             oppostfix:=PF_SB;           OS_16:             oppostfix:=PF_H;           OS_S16:             oppostfix:=PF_SH;           OS_32,           OS_S32:             oppostfix:=PF_None;           else             InternalError(200308298);         end;         if (ref.alignment in [1,2]) and (ref.alignment<tcgsize2size[fromsize]) then           begin             if target_info.endian=endian_big then               dir:=-1             else               dir:=1;             case FromSize of               OS_16,OS_S16:                 begin                   { only complicated references need an extra loadaddr }                   if assigned(ref.symbol) or                     (ref.index<>NR_NO) or                     (ref.offset<-124) or                     (ref.offset>124) or                     { sometimes the compiler reused registers }                     (reg=ref.index) or                     (reg=ref.base) then                     begin                       tmpreg2:=getintregister(list,OS_INT);                       a_loadaddr_ref_reg(list,ref,tmpreg2);                       reference_reset_base(usedtmpref,tmpreg2,0,ref.alignment);                     end                   else                     usedtmpref:=ref;                   if target_info.endian=endian_big then                     inc(usedtmpref.offset,1);                   tmpreg:=getintregister(list,OS_INT);                   a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,reg);                   inc(usedtmpref.offset,dir);                   if FromSize=OS_16 then                     a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg)                   else                     a_internal_load_ref_reg(list,OS_S8,OS_S8,usedtmpref,tmpreg);                   list.concat(taicpu.op_reg_const(A_LSL,tmpreg,8));                   list.concat(taicpu.op_reg_reg(A_ORR,reg,tmpreg));                 end;               OS_32,OS_S32:                 begin                   tmpreg:=getintregister(list,OS_INT);                   { only complicated references need an extra loadaddr }                   if assigned(ref.symbol) or                     (ref.index<>NR_NO) or                     (ref.offset<-124) or                     (ref.offset>124) or                     { sometimes the compiler reused registers }                     (reg=ref.index) or                     (reg=ref.base) then                     begin                       tmpreg2:=getintregister(list,OS_INT);                       a_loadaddr_ref_reg(list,ref,tmpreg2);                       reference_reset_base(usedtmpref,tmpreg2,0,ref.alignment);                     end                   else                     usedtmpref:=ref;                   if ref.alignment=2 then                     begin                       if target_info.endian=endian_big then                         inc(usedtmpref.offset,2);                       a_internal_load_ref_reg(list,OS_16,OS_16,usedtmpref,reg);                       inc(usedtmpref.offset,dir*2);                       a_internal_load_ref_reg(list,OS_16,OS_16,usedtmpref,tmpreg);                       list.concat(taicpu.op_reg_const(A_LSL,tmpreg,16));                       list.concat(taicpu.op_reg_reg(A_ORR,reg,tmpreg));                     end                   else                     begin                       if target_info.endian=endian_big then                         inc(usedtmpref.offset,3);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,reg);                       inc(usedtmpref.offset,dir);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                       list.concat(taicpu.op_reg_const(A_LSL,tmpreg,8));                       list.concat(taicpu.op_reg_reg(A_ORR,reg,tmpreg));                       inc(usedtmpref.offset,dir);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                       list.concat(taicpu.op_reg_const(A_LSL,tmpreg,16));                       list.concat(taicpu.op_reg_reg(A_ORR,reg,tmpreg));                       inc(usedtmpref.offset,dir);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                       list.concat(taicpu.op_reg_const(A_LSL,tmpreg,24));                       list.concat(taicpu.op_reg_reg(A_ORR,reg,tmpreg));                     end;                 end               else                 handle_load_store(list,A_LDR,oppostfix,reg,ref);             end;           end         else           handle_load_store(list,A_LDR,oppostfix,reg,ref);         if (fromsize=OS_S8) and (tosize = OS_16) then           a_load_reg_reg(list,OS_16,OS_32,reg,reg);       end;     procedure tthumbcgarm.a_load_const_reg(list : TAsmList; size: tcgsize; a : tcgint;reg : tregister);       var          imm_shift : byte;          l : tasmlabel;          hr : treference;       begin          if not(size in [OS_8,OS_S8,OS_16,OS_S16,OS_32,OS_S32]) then            internalerror(2002090902);          if is_thumb_imm(a) then            list.concat(taicpu.op_reg_const(A_MOV,reg,a))          else            begin              reference_reset(hr,4);              current_asmdata.getjumplabel(l);              cg.a_label(current_procinfo.aktlocaldata,l);              hr.symboldata:=current_procinfo.aktlocaldata.last;              current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(longint(a)));              hr.symbol:=l;              hr.base:=NR_PC;              list.concat(taicpu.op_reg_ref(A_LDR,reg,hr));            end;       end;    procedure tthumbcgarm.g_adjust_self_value(list:TAsmList;procdef: tprocdef;ioffset: tcgint);      var        hsym : tsym;        href,        tmpref : treference;        paraloc : Pcgparalocation;        l : TAsmLabel;      begin        { calculate the parameter info for the procdef }        procdef.init_paraloc_info(callerside);        hsym:=tsym(procdef.parast.Find('self'));        if not(assigned(hsym) and          (hsym.typ=paravarsym)) then          internalerror(200305251);        paraloc:=tparavarsym(hsym).paraloc[callerside].location;        while paraloc<>nil do          with paraloc^ do            begin              case loc of                LOC_REGISTER:                  begin                    if is_thumb_imm(ioffset) then                      a_op_const_reg(list,OP_SUB,size,ioffset,register)                    else                      begin                        list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,[RS_R4]));                        reference_reset(tmpref,4);                        current_asmdata.getjumplabel(l);                        current_procinfo.aktlocaldata.Concat(tai_align.Create(4));                        cg.a_label(current_procinfo.aktlocaldata,l);                        tmpref.symboldata:=current_procinfo.aktlocaldata.last;                        current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(ioffset));                        tmpref.symbol:=l;                        tmpref.base:=NR_PC;                        list.concat(taicpu.op_reg_ref(A_LDR,NR_R4,tmpref));                        a_op_reg_reg(list,OP_SUB,size,NR_R4,register);                        list.concat(taicpu.op_regset(A_POP,R_INTREGISTER,R_SUBWHOLE,[RS_R4]));                      end;                  end;                LOC_REFERENCE:                  begin                    { offset in the wrapper needs to be adjusted for the stored                      return address }                    reference_reset_base(href,reference.index,reference.offset+sizeof(aint),sizeof(pint));                    if is_thumb_imm(ioffset) then                      a_op_const_ref(list,OP_SUB,size,ioffset,href)                    else                      begin                        list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,[RS_R4]));                        reference_reset(tmpref,4);                        current_asmdata.getjumplabel(l);                        current_procinfo.aktlocaldata.Concat(tai_align.Create(4));                        cg.a_label(current_procinfo.aktlocaldata,l);                        tmpref.symboldata:=current_procinfo.aktlocaldata.last;                        current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(ioffset));                        tmpref.symbol:=l;                        tmpref.base:=NR_PC;                        list.concat(taicpu.op_reg_ref(A_LDR,NR_R4,tmpref));                        a_op_reg_ref(list,OP_SUB,size,NR_R4,href);                        list.concat(taicpu.op_regset(A_POP,R_INTREGISTER,R_SUBWHOLE,[RS_R4]));                      end;                  end                else                  internalerror(200309189);              end;              paraloc:=next;            end;      end;    function tthumbcgarm.handle_load_store(list: TAsmList; op: tasmop; oppostfix: toppostfix; reg: tregister; ref: treference): treference;      var        href : treference;        tmpreg : TRegister;      begin        href:=ref;        if { LDR/STR limitations }           (            (((op=A_LDR) and (oppostfix=PF_None)) or             ((op=A_STR) and (oppostfix=PF_None))) and            (ref.base<>NR_STACK_POINTER_REG) and            (abs(ref.offset)>124)           ) or           { LDRB/STRB limitations }           (           (((op=A_LDR) and (oppostfix=PF_B)) or            ((op=A_LDRB) and (oppostfix=PF_None)) or            ((op=A_STR) and (oppostfix=PF_B)) or            ((op=A_STRB) and (oppostfix=PF_None))) and            ((ref.base=NR_STACK_POINTER_REG) or             (ref.index=NR_STACK_POINTER_REG) or             (abs(ref.offset)>31)            )           ) or           { LDRH/STRH limitations }           (            (((op=A_LDR) and (oppostfix=PF_H)) or             ((op=A_LDRH) and (oppostfix=PF_None)) or             ((op=A_STR) and (oppostfix=PF_H)) or             ((op=A_STRH) and (oppostfix=PF_None))) and            ((ref.base=NR_STACK_POINTER_REG) or             (ref.index=NR_STACK_POINTER_REG) or             (abs(ref.offset)>62) or             ((abs(ref.offset) mod 2)<>0)            )           ) then          begin            tmpreg:=getintregister(list,OS_ADDR);            a_loadaddr_ref_reg(list,ref,tmpreg);            reference_reset_base(href,tmpreg,0,ref.alignment);          end        else if (op=A_LDR) and           (oppostfix in [PF_None]) and           (ref.base=NR_STACK_POINTER_REG) and           (abs(ref.offset)>1020) then          begin            tmpreg:=getintregister(list,OS_ADDR);            a_loadaddr_ref_reg(list,ref,tmpreg);            reference_reset_base(href,tmpreg,0,ref.alignment);          end        else if (op=A_LDR) and           ((oppostfix in [PF_SH,PF_SB]) or            (abs(ref.offset)>124)) then          begin            tmpreg:=getintregister(list,OS_ADDR);            a_loadaddr_ref_reg(list,ref,tmpreg);            reference_reset_base(href,tmpreg,0,ref.alignment);          end;        Result:=inherited handle_load_store(list, op, oppostfix, reg, href);      end;    procedure tthumbcgarm.a_op_reg_reg(list : TAsmList; Op: TOpCG; size: TCGSize; src, dst: TRegister);      var        tmpreg,overflowreg : tregister;        asmop : tasmop;      begin        case op of          OP_NEG:            list.concat(taicpu.op_reg_reg(A_NEG,dst,src));          OP_NOT:            list.concat(taicpu.op_reg_reg(A_MVN,dst,src));          OP_DIV,OP_IDIV:            internalerror(200308284);          OP_ROL:            begin              if not(size in [OS_32,OS_S32]) then                internalerror(2008072801);              { simulate ROL by ror'ing 32-value }              tmpreg:=getintregister(list,OS_32);              a_load_const_reg(list,OS_32,32,tmpreg);              list.concat(taicpu.op_reg_reg(A_SUB,tmpreg,src));              list.concat(taicpu.op_reg_reg(A_ROR,dst,src));            end;          else            begin              a_reg_alloc(list,NR_DEFAULTFLAGS);              list.concat(setoppostfix(                taicpu.op_reg_reg(op_reg_opcg2asmop[op],dst,src),op_reg_postfix[op]));            end;        end;        maybeadjustresult(list,op,size,dst);      end;    procedure tthumbcgarm.a_op_const_reg(list: TAsmList; op: TOpCg; size: tcgsize; a: tcgint; dst: tregister);      var        tmpreg : tregister;        so : tshifterop;        l1 : longint;        imm1, imm2: DWord;      begin        //!!! ovloc.loc:=LOC_VOID;        if {$ifopt R+}(a<>-2147483648) and{$endif} {!!!!!! not setflags and } is_thumb_imm(-a) then          case op of            OP_ADD:              begin                op:=OP_SUB;                a:=aint(dword(-a));              end;            OP_SUB:              begin                op:=OP_ADD;                a:=aint(dword(-a));              end          end;        if is_thumb_imm(a) and (op in [OP_ADD,OP_SUB]) then          begin             // if cgsetflags or setflags then             a_reg_alloc(list,NR_DEFAULTFLAGS);            list.concat(setoppostfix(              taicpu.op_reg_const(op_reg_opcg2asmop[op],dst,a),op_reg_postfix[op]));            if (cgsetflags {!!! or setflags }) and (size in [OS_8,OS_16,OS_32]) then              begin                //!!! ovloc.loc:=LOC_FLAGS;                case op of                  OP_ADD:                    //!!! ovloc.resflags:=F_CS;                    ;                  OP_SUB:                    //!!! ovloc.resflags:=F_CC;                    ;                end;              end;          end        else          begin            { there could be added some more sophisticated optimizations }            if (op in [OP_MUL,OP_IMUL,OP_DIV,OP_IDIV]) and (a=1) then              a_load_reg_reg(list,size,size,dst,dst)            else if (op in [OP_MUL,OP_IMUL]) and (a=0) then              a_load_const_reg(list,size,0,dst)            else if (op in [OP_IMUL,OP_IDIV]) and (a=-1) then              a_op_reg_reg(list,OP_NEG,size,dst,dst)            { we do this here instead in the peephole optimizer because              it saves us a register }{$ifdef DUMMY}            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a,l1) and not(cgsetflags or setflags) then              a_op_const_reg_reg(list,OP_SHL,size,l1,dst,dst)            { for example : b=a*5 -> b=a*4+a with add instruction and shl }            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a-1,l1) and not(cgsetflags or setflags) then              begin                if l1>32 then{roozbeh does this ever happen?}                  internalerror(200308296);                shifterop_reset(so);                so.shiftmode:=SM_LSL;                so.shiftimm:=l1;                list.concat(taicpu.op_reg_reg_reg_shifterop(A_ADD,dst,dst,dst,so));              end            { for example : b=a*7 -> b=a*8-a with rsb instruction and shl }            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a+1,l1) and not(cgsetflags or setflags) then              begin                if l1>32 then{does this ever happen?}                  internalerror(201205181);                shifterop_reset(so);                so.shiftmode:=SM_LSL;                so.shiftimm:=l1;                list.concat(taicpu.op_reg_reg_reg_shifterop(A_RSB,dst,dst,dst,so));              end            else if (op in [OP_MUL,OP_IMUL]) and not(cgsetflags or setflags) and try_optimized_mul32_const_reg_reg(list,a,dst,dst) then              begin                { nothing to do on success }              end{$endif DUMMY}            { x := y and 0; just clears a register, this sometimes gets generated on 64bit ops.              Just using mov x, #0 might allow some easier optimizations down the line. }            else if (op = OP_AND) and (dword(a)=0) then              list.concat(taicpu.op_reg_const(A_MOV,dst,0))            { x := y AND $FFFFFFFF just copies the register, so use mov for better optimizations }            else if (op = OP_AND) and (not(dword(a))=0) then              // do nothing            { BIC clears the specified bits, while AND keeps them, using BIC allows to use a              broader range of shifterconstants.}{$ifdef DUMMY}            else if (op = OP_AND) and is_shifter_const(not(dword(a)),shift) then              list.concat(taicpu.op_reg_reg_const(A_BIC,dst,dst,not(dword(a))))            else if (op = OP_AND) and split_into_shifter_const(not(dword(a)), imm1, imm2) then              begin                list.concat(taicpu.op_reg_reg_const(A_BIC,dst,dst,imm1));                list.concat(taicpu.op_reg_reg_const(A_BIC,dst,dst,imm2));              end            else if (op in [OP_ADD, OP_SUB, OP_OR]) and                    not(cgsetflags or setflags) and                    split_into_shifter_const(a, imm1, imm2) then              begin                list.concat(taicpu.op_reg_reg_const(op_reg_reg_opcg2asmop[op],dst,dst,imm1));                list.concat(taicpu.op_reg_reg_const(op_reg_reg_opcg2asmop[op],dst,dst,imm2));              end{$endif DUMMY}            else if (op in [OP_SHL, OP_SHR, OP_SAR]) then              begin                list.concat(taicpu.op_reg_reg_const(op_reg_opcg2asmop[op],dst,dst,a));              end            else              begin                tmpreg:=getintregister(list,size);                a_load_const_reg(list,size,a,tmpreg);                a_op_reg_reg(list,op,size,tmpreg,dst);              end;          end;        maybeadjustresult(list,op,size,dst);      end;    procedure tthumbcgarm.a_op_const_reg_reg(list: TAsmList; op: TOpCg; size: tcgsize; a: tcgint; src, dst: tregister);      begin        if (op=OP_ADD) and (src=NR_R13) and (dst<>NR_R13) and ((a mod 4)=0) and (a>0) and (a<=1020) then          list.concat(taicpu.op_reg_reg_const(A_ADD,dst,src,a))        else          inherited a_op_const_reg_reg(list,op,size,a,src,dst);      end;    procedure tthumbcgarm.g_flags2reg(list: TAsmList; size: TCgSize; const f: TResFlags; reg: TRegister);      var        l1,l2 : tasmlabel;        ai : taicpu;      begin        current_asmdata.getjumplabel(l1);        current_asmdata.getjumplabel(l2);        ai:=setcondition(taicpu.op_sym(A_B,l1),flags_to_cond(f));        ai.is_jmp:=true;        list.concat(ai);        list.concat(taicpu.op_reg_const(A_MOV,reg,0));        list.concat(taicpu.op_sym(A_B,l2));        cg.a_label(list,l1);        list.concat(taicpu.op_reg_const(A_MOV,reg,1));        a_reg_dealloc(list,NR_DEFAULTFLAGS);        cg.a_label(list,l2);      end;    procedure tthumbcgarm.g_external_wrapper(list: TAsmList; procdef: tprocdef; const externalname: string);      var        tmpref : treference;        l : tasmlabel;      begin        { there is no branch instruction on thumb which allows big distances and which leaves LR as it is          and which allows to switch the instruction set }        { create const entry }        reference_reset(tmpref,4);        current_asmdata.getjumplabel(l);        tmpref.symbol:=l;        tmpref.base:=NR_PC;        list.concat(taicpu.op_regset(A_PUSH,R_INTREGISTER,R_SUBWHOLE,[RS_R0]));        list.concat(taicpu.op_reg_ref(A_LDR,NR_R0,tmpref));        list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_R0));        list.concat(taicpu.op_regset(A_POP,R_INTREGISTER,R_SUBWHOLE,[RS_R0]));        list.concat(taicpu.op_reg(A_BX,NR_R12));        { append const entry }        list.Concat(tai_align.Create(4));        list.Concat(tai_label.create(l));        list.concat(tai_const.Create_sym(current_asmdata.RefAsmSymbol(externalname)));      end;    procedure tthumb2cgarm.init_register_allocators;      begin        inherited init_register_allocators;        { currently, we save R14 always, so we can use it }        if (target_info.system<>system_arm_darwin) then          rg[R_INTREGISTER]:=trgintcputhumb2.create(R_INTREGISTER,R_SUBWHOLE,              [RS_R0,RS_R1,RS_R2,RS_R3,RS_R4,RS_R5,RS_R6,RS_R7,RS_R8,               RS_R9,RS_R10,RS_R12,RS_R14],first_int_imreg,[])        else          { r9 is not available on Darwin according to the llvm code generator }          rg[R_INTREGISTER]:=trgintcputhumb2.create(R_INTREGISTER,R_SUBWHOLE,              [RS_R0,RS_R1,RS_R2,RS_R3,RS_R4,RS_R5,RS_R6,RS_R7,RS_R8,               RS_R10,RS_R12,RS_R14],first_int_imreg,[]);        rg[R_FPUREGISTER]:=trgcpu.create(R_FPUREGISTER,R_SUBNONE,            [RS_F0,RS_F1,RS_F2,RS_F3,RS_F4,RS_F5,RS_F6,RS_F7],first_fpu_imreg,[]);        if current_settings.fputype in [fpu_fpv4_s16,fpu_vfpv3_d16] then          rg[R_MMREGISTER]:=trgcpu.create(R_MMREGISTER,R_SUBFD,              [RS_D0,RS_D1,RS_D2,RS_D3,RS_D4,RS_D5,RS_D6,RS_D7,               RS_D8,RS_D9,RS_D10,RS_D11,RS_D12,RS_D13,RS_D14,RS_D15              ],first_mm_imreg,[])        else          rg[R_MMREGISTER]:=trgcpu.create(R_MMREGISTER,R_SUBNONE,              [RS_S0,RS_S1,RS_R2,RS_R3,RS_R4,RS_S31],first_mm_imreg,[]);      end;    procedure tthumb2cgarm.done_register_allocators;      begin        rg[R_INTREGISTER].free;        rg[R_FPUREGISTER].free;        rg[R_MMREGISTER].free;        inherited done_register_allocators;      end;    procedure tthumb2cgarm.a_call_reg(list : TAsmList;reg: tregister);      begin        list.concat(taicpu.op_reg(A_BLX, reg));{        the compiler does not properly set this flag anymore in pass 1, and        for now we only need it after pass 2 (I hope) (JM)          if not(pi_do_call in current_procinfo.flags) then            internalerror(2003060703);}        include(current_procinfo.flags,pi_do_call);      end;     procedure tthumb2cgarm.a_load_const_reg(list : TAsmList; size: tcgsize; a : tcgint;reg : tregister);       var          imm_shift : byte;          l : tasmlabel;          hr : treference;       begin          if not(size in [OS_8,OS_S8,OS_16,OS_S16,OS_32,OS_S32]) then            internalerror(2002090902);          if is_thumb32_imm(a) then            list.concat(taicpu.op_reg_const(A_MOV,reg,a))          else if is_thumb32_imm(not(a)) then            list.concat(taicpu.op_reg_const(A_MVN,reg,not(a)))          else if (a and $FFFF)=a then            list.concat(taicpu.op_reg_const(A_MOVW,reg,a))          else            begin               reference_reset(hr,4);               current_asmdata.getjumplabel(l);               cg.a_label(current_procinfo.aktlocaldata,l);               hr.symboldata:=current_procinfo.aktlocaldata.last;               current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(longint(a)));               hr.symbol:=l;               hr.base:=NR_PC;               list.concat(taicpu.op_reg_ref(A_LDR,reg,hr));            end;       end;     procedure tthumb2cgarm.a_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister);       var         oppostfix:toppostfix;         usedtmpref: treference;         tmpreg,tmpreg2 : tregister;         so : tshifterop;         dir : integer;       begin         if (TCGSize2Size[FromSize] >= TCGSize2Size[ToSize]) then           FromSize := ToSize;         case FromSize of           { signed integer registers }           OS_8:             oppostfix:=PF_B;           OS_S8:             oppostfix:=PF_SB;           OS_16:             oppostfix:=PF_H;           OS_S16:             oppostfix:=PF_SH;           OS_32,           OS_S32:             oppostfix:=PF_None;           else             InternalError(200308299);         end;         if (ref.alignment in [1,2]) and (ref.alignment<tcgsize2size[fromsize]) then           begin             if target_info.endian=endian_big then               dir:=-1             else               dir:=1;             case FromSize of               OS_16,OS_S16:                 begin                   { only complicated references need an extra loadaddr }                   if assigned(ref.symbol) or                     (ref.index<>NR_NO) or                     (ref.offset<-255) or                     (ref.offset>4094) or                     { sometimes the compiler reused registers }                     (reg=ref.index) or                     (reg=ref.base) then                     begin                       tmpreg2:=getintregister(list,OS_INT);                       a_loadaddr_ref_reg(list,ref,tmpreg2);                       reference_reset_base(usedtmpref,tmpreg2,0,ref.alignment);                     end                   else                     usedtmpref:=ref;                   if target_info.endian=endian_big then                     inc(usedtmpref.offset,1);                   shifterop_reset(so);so.shiftmode:=SM_LSL;so.shiftimm:=8;                   tmpreg:=getintregister(list,OS_INT);                   a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,reg);                   inc(usedtmpref.offset,dir);                   if FromSize=OS_16 then                     a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg)                   else                     a_internal_load_ref_reg(list,OS_S8,OS_S8,usedtmpref,tmpreg);                   list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg,so));                 end;               OS_32,OS_S32:                 begin                   tmpreg:=getintregister(list,OS_INT);                   { only complicated references need an extra loadaddr }                   if assigned(ref.symbol) or                     (ref.index<>NR_NO) or                     (ref.offset<-255) or                     (ref.offset>4092) or                     { sometimes the compiler reused registers }                     (reg=ref.index) or                     (reg=ref.base) then                     begin                       tmpreg2:=getintregister(list,OS_INT);                       a_loadaddr_ref_reg(list,ref,tmpreg2);                       reference_reset_base(usedtmpref,tmpreg2,0,ref.alignment);                     end                   else                     usedtmpref:=ref;                   shifterop_reset(so);so.shiftmode:=SM_LSL;                   if ref.alignment=2 then                     begin                       if target_info.endian=endian_big then                         inc(usedtmpref.offset,2);                       a_internal_load_ref_reg(list,OS_16,OS_16,usedtmpref,reg);                       inc(usedtmpref.offset,dir*2);                       a_internal_load_ref_reg(list,OS_16,OS_16,usedtmpref,tmpreg);                       so.shiftimm:=16;                       list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg,so));                     end                   else                     begin                       if target_info.endian=endian_big then                         inc(usedtmpref.offset,3);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,reg);                       inc(usedtmpref.offset,dir);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                       so.shiftimm:=8;                       list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg,so));                       inc(usedtmpref.offset,dir);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                       so.shiftimm:=16;                       list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg,so));                       inc(usedtmpref.offset,dir);                       a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);                       so.shiftimm:=24;                       list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,reg,tmpreg,so));                     end;                 end               else                 handle_load_store(list,A_LDR,oppostfix,reg,ref);             end;           end         else           handle_load_store(list,A_LDR,oppostfix,reg,ref);         if (fromsize=OS_S8) and (tosize = OS_16) then           a_load_reg_reg(list,OS_16,OS_32,reg,reg);       end;    procedure tthumb2cgarm.a_op_reg_reg(list : TAsmList; Op: TOpCG; size: TCGSize; src, dst: TRegister);      begin        if op = OP_NOT then          begin            list.concat(taicpu.op_reg_reg(A_MVN,dst,src));            case size of              OS_8: list.concat(taicpu.op_reg_reg(A_UXTB,dst,dst));              OS_S8: list.concat(taicpu.op_reg_reg(A_SXTB,dst,dst));              OS_16: list.concat(taicpu.op_reg_reg(A_UXTH,dst,dst));              OS_S16: list.concat(taicpu.op_reg_reg(A_SXTH,dst,dst));            end;          end        else          inherited a_op_reg_reg(list, op, size, src, dst);      end;    procedure tthumb2cgarm.a_op_const_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; a: tcgint; src, dst: tregister;setflags : boolean;var ovloc : tlocation);      var        shift, width : byte;        tmpreg : tregister;        so : tshifterop;        l1 : longint;      begin        ovloc.loc:=LOC_VOID;        if {$ifopt R+}(a<>-2147483648) and{$endif} is_shifter_const(-a,shift) then          case op of            OP_ADD:              begin                op:=OP_SUB;                a:=aint(dword(-a));              end;            OP_SUB:              begin                op:=OP_ADD;                a:=aint(dword(-a));              end          end;        if is_shifter_const(a,shift) and not(op in [OP_IMUL,OP_MUL]) then          case op of            OP_NEG,OP_NOT,            OP_DIV,OP_IDIV:              internalerror(200308285);            OP_SHL:              begin                if a>32 then                  internalerror(2014020703);                if a<>0 then                  begin                    shifterop_reset(so);                    so.shiftmode:=SM_LSL;                    so.shiftimm:=a;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                  end                else                 list.concat(taicpu.op_reg_reg(A_MOV,dst,src));              end;            OP_ROL:              begin                if a>32 then                  internalerror(2014020704);                if a<>0 then                  begin                    shifterop_reset(so);                    so.shiftmode:=SM_ROR;                    so.shiftimm:=32-a;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                  end                else                 list.concat(taicpu.op_reg_reg(A_MOV,dst,src));              end;            OP_ROR:              begin                if a>32 then                  internalerror(2014020705);                if a<>0 then                  begin                    shifterop_reset(so);                    so.shiftmode:=SM_ROR;                    so.shiftimm:=a;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                  end                else                 list.concat(taicpu.op_reg_reg(A_MOV,dst,src));              end;            OP_SHR:              begin                if a>32 then                  internalerror(200308292);                shifterop_reset(so);                if a<>0 then                  begin                    so.shiftmode:=SM_LSR;                    so.shiftimm:=a;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                  end                else                 list.concat(taicpu.op_reg_reg(A_MOV,dst,src));              end;            OP_SAR:              begin                if a>32 then                  internalerror(200308295);                if a<>0 then                  begin                    shifterop_reset(so);                    so.shiftmode:=SM_ASR;                    so.shiftimm:=a;                    list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));                  end                else                 list.concat(taicpu.op_reg_reg(A_MOV,dst,src));              end;            else              if (op in [OP_SUB, OP_ADD]) and                 ((a < 0) or                  (a > 4095)) then                begin                  tmpreg:=getintregister(list,size);                  a_load_const_reg(list, size, a, tmpreg);                  if cgsetflags or setflags then                    a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(setoppostfix(                    taicpu.op_reg_reg_reg(op_reg_reg_opcg2asmop[op],dst,src,tmpreg),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))));                end              else                begin                  if cgsetflags or setflags then                    a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(setoppostfix(                    taicpu.op_reg_reg_const(op_reg_reg_opcg2asmop[op],dst,src,a),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))));                end;              if (cgsetflags or setflags) and (size in [OS_8,OS_16,OS_32]) then                begin                  ovloc.loc:=LOC_FLAGS;                  case op of                    OP_ADD:                      ovloc.resflags:=F_CS;                    OP_SUB:                      ovloc.resflags:=F_CC;                  end;                end;          end        else          begin            { there could be added some more sophisticated optimizations }            if (op in [OP_MUL,OP_IMUL]) and (a=1) then              a_load_reg_reg(list,size,size,src,dst)            else if (op in [OP_MUL,OP_IMUL]) and (a=0) then              a_load_const_reg(list,size,0,dst)            else if (op in [OP_IMUL]) and (a=-1) then              a_op_reg_reg(list,OP_NEG,size,src,dst)            { we do this here instead in the peephole optimizer because              it saves us a register }            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a,l1) and not(cgsetflags or setflags) then              a_op_const_reg_reg(list,OP_SHL,size,l1,src,dst)            { for example : b=a*5 -> b=a*4+a with add instruction and shl }            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a-1,l1) and not(cgsetflags or setflags) then              begin                if l1>32 then{roozbeh does this ever happen?}                  internalerror(200308296);                shifterop_reset(so);                so.shiftmode:=SM_LSL;                so.shiftimm:=l1;                list.concat(taicpu.op_reg_reg_reg_shifterop(A_ADD,dst,src,src,so));              end            { for example : b=a*7 -> b=a*8-a with rsb instruction and shl }            else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a+1,l1) and not(cgsetflags or setflags) then              begin                if l1>32 then{does this ever happen?}                  internalerror(201205181);                shifterop_reset(so);                so.shiftmode:=SM_LSL;                so.shiftimm:=l1;                list.concat(taicpu.op_reg_reg_reg_shifterop(A_RSB,dst,src,src,so));              end            else if (op in [OP_MUL,OP_IMUL]) and not(cgsetflags or setflags) and try_optimized_mul32_const_reg_reg(list,a,src,dst) then              begin                { nothing to do on success }              end            { x := y and 0; just clears a register, this sometimes gets generated on 64bit ops.              Just using mov x, #0 might allow some easier optimizations down the line. }            else if (op = OP_AND) and (dword(a)=0) then              list.concat(taicpu.op_reg_const(A_MOV,dst,0))            { x := y AND $FFFFFFFF just copies the register, so use mov for better optimizations }            else if (op = OP_AND) and (not(dword(a))=0) then              list.concat(taicpu.op_reg_reg(A_MOV,dst,src))            { BIC clears the specified bits, while AND keeps them, using BIC allows to use a              broader range of shifterconstants.}            {else if (op = OP_AND) and is_shifter_const(not(dword(a)),shift) then              list.concat(taicpu.op_reg_reg_const(A_BIC,dst,src,not(dword(a))))}            else if (op = OP_AND) and is_thumb32_imm(a) then              list.concat(taicpu.op_reg_reg_const(A_AND,dst,src,dword(a)))            else if (op = OP_AND) and (a = $FFFF) then              list.concat(taicpu.op_reg_reg(A_UXTH,dst,src))            else if (op = OP_AND) and is_thumb32_imm(not(dword(a))) then              list.concat(taicpu.op_reg_reg_const(A_BIC,dst,src,not(dword(a))))            else if (op = OP_AND) and is_continuous_mask(not(a), shift, width) then              begin                a_load_reg_reg(list,size,size,src,dst);                list.concat(taicpu.op_reg_const_const(A_BFC,dst,shift,width))              end            else              begin                tmpreg:=getintregister(list,size);                a_load_const_reg(list,size,a,tmpreg);                a_op_reg_reg_reg_checkoverflow(list,op,size,tmpreg,src,dst,setflags,ovloc);              end;          end;        maybeadjustresult(list,op,size,dst);      end;    const      op_reg_reg_opcg2asmopThumb2: array[TOpCG] of tasmop =        (A_NONE,A_MOV,A_ADD,A_AND,A_UDIV,A_SDIV,A_MUL,A_MUL,A_NONE,A_MVN,A_ORR,         A_ASR,A_LSL,A_LSR,A_SUB,A_EOR,A_NONE,A_ROR);    procedure tthumb2cgarm.a_op_reg_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; src1, src2, dst: tregister;setflags : boolean;var ovloc : tlocation);      var        so : tshifterop;        tmpreg,overflowreg : tregister;        asmop : tasmop;      begin        ovloc.loc:=LOC_VOID;        case op of           OP_NEG,OP_NOT:              internalerror(200308286);           OP_ROL:              begin                if not(size in [OS_32,OS_S32]) then                   internalerror(2008072801);                { simulate ROL by ror'ing 32-value }                tmpreg:=getintregister(list,OS_32);                list.concat(taicpu.op_reg_const(A_MOV,tmpreg,32));                list.concat(taicpu.op_reg_reg_reg(A_SUB,src1,tmpreg,src1));                list.concat(taicpu.op_reg_reg_reg(A_ROR, dst, src2, src1));              end;           OP_ROR:              begin                if not(size in [OS_32,OS_S32]) then                   internalerror(2008072802);                list.concat(taicpu.op_reg_reg_reg(A_ROR, dst, src2, src1));              end;           OP_IMUL,           OP_MUL:              begin                if cgsetflags or setflags then                   begin                      overflowreg:=getintregister(list,size);                      if op=OP_IMUL then                        asmop:=A_SMULL                      else                        asmop:=A_UMULL;                      { the arm doesn't allow that rd and rm are the same }                      if dst=src2 then                        begin                           if dst<>src1 then                              list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,src1,src2))                           else                              begin                                tmpreg:=getintregister(list,size);                                a_load_reg_reg(list,size,size,src2,dst);                                list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,tmpreg,src1));                              end;                        end                      else                        list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,src2,src1));                      a_reg_alloc(list,NR_DEFAULTFLAGS);                      if op=OP_IMUL then                        begin                           shifterop_reset(so);                           so.shiftmode:=SM_ASR;                           so.shiftimm:=31;                           list.concat(taicpu.op_reg_reg_shifterop(A_CMP,overflowreg,dst,so));                        end                      else                        list.concat(taicpu.op_reg_const(A_CMP,overflowreg,0));                       ovloc.loc:=LOC_FLAGS;                       ovloc.resflags:=F_NE;                   end                else                   begin                      { the arm doesn't allow that rd and rm are the same }                      if dst=src2 then                        begin                           if dst<>src1 then                              list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,src1,src2))                           else                              begin                                tmpreg:=getintregister(list,size);                                a_load_reg_reg(list,size,size,src2,dst);                                list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,tmpreg,src1));                              end;                        end                      else                        list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,src2,src1));                   end;              end;           else             begin               if cgsetflags or setflags then                 a_reg_alloc(list,NR_DEFAULTFLAGS);{$ifdef dummy}               { R13 is not allowed for certain instruction operands }               if op_reg_reg_opcg2asmopThumb2[op] in [A_ADD,A_SUB,A_AND,A_BIC,A_EOR] then                 begin                   if getsupreg(dst)=RS_R13 then                     begin                       tmpreg:=getintregister(list,OS_INT);                       a_load_reg_reg(list,OS_INT,OS_INT,dst,tmpreg);                       dst:=tmpreg;                     end;                   if getsupreg(src1)=RS_R13 then                     begin                       tmpreg:=getintregister(list,OS_INT);                       a_load_reg_reg(list,OS_INT,OS_INT,src1,tmpreg);                       src1:=tmpreg;                     end;                 end;{$endif}               list.concat(setoppostfix(                 taicpu.op_reg_reg_reg(op_reg_reg_opcg2asmopThumb2[op],dst,src2,src1),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))));             end;        end;        maybeadjustresult(list,op,size,dst);      end;    procedure tthumb2cgarm.g_flags2reg(list: TAsmList; size: TCgSize; const f: TResFlags; reg: TRegister);      var item: taicpu;      begin        list.concat(taicpu.op_cond(A_ITE, flags_to_cond(f)));        list.concat(setcondition(taicpu.op_reg_const(A_MOV,reg,1),flags_to_cond(f)));        list.concat(setcondition(taicpu.op_reg_const(A_MOV,reg,0),inverse_cond(flags_to_cond(f))));      end;    procedure tthumb2cgarm.g_proc_entry(list : TAsmList;localsize : longint;nostackframe:boolean);      var         ref : treference;         shift : byte;         firstfloatreg,lastfloatreg,         r : byte;         regs : tcpuregisterset;         stackmisalignment: pint;      begin        LocalSize:=align(LocalSize,4);        { call instruction does not put anything on the stack }        stackmisalignment:=0;        if not(nostackframe) then          begin            firstfloatreg:=RS_NO;            lastfloatreg:=RS_NO;            { save floating point registers? }            for r:=RS_F0 to RS_F7 do              if r in rg[R_FPUREGISTER].used_in_proc-paramanager.get_volatile_registers_fpu(pocall_stdcall) then                begin                  if firstfloatreg=RS_NO then                    firstfloatreg:=r;                  lastfloatreg:=r;                  inc(stackmisalignment,12);                end;            a_reg_alloc(list,NR_STACK_POINTER_REG);            if current_procinfo.framepointer<>NR_STACK_POINTER_REG then              begin                a_reg_alloc(list,NR_FRAME_POINTER_REG);                a_reg_alloc(list,NR_R12);                list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_STACK_POINTER_REG));              end;            { save int registers }            reference_reset(ref,4);            ref.index:=NR_STACK_POINTER_REG;            ref.addressmode:=AM_PREINDEXED;            regs:=rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall);            if current_procinfo.framepointer<>NR_STACK_POINTER_REG then              regs:=regs+[RS_FRAME_POINTER_REG,RS_R14]            else if (regs<>[]) or (pi_do_call in current_procinfo.flags) then              include(regs,RS_R14);            if regs<>[] then              begin                for r:=RS_R0 to RS_R15 do                  if (r in regs) then                    inc(stackmisalignment,4);                list.concat(setoppostfix(taicpu.op_ref_regset(A_STM,ref,R_INTREGISTER,R_SUBWHOLE,regs),PF_FD));              end;            if current_procinfo.framepointer<>NR_STACK_POINTER_REG then              begin                { the framepointer now points to the saved R15, so the saved                  framepointer is at R11-12 (for get_caller_frame) }                list.concat(taicpu.op_reg_reg_const(A_SUB,NR_FRAME_POINTER_REG,NR_R12,4));                a_reg_dealloc(list,NR_R12);              end;            stackmisalignment:=stackmisalignment mod current_settings.alignment.localalignmax;            if (LocalSize<>0) or               ((stackmisalignment<>0) and                ((pi_do_call in current_procinfo.flags) or                 (po_assembler in current_procinfo.procdef.procoptions))) then              begin                localsize:=align(localsize+stackmisalignment,current_settings.alignment.localalignmax)-stackmisalignment;                if not(is_shifter_const(localsize,shift)) then                  begin                    if current_procinfo.framepointer=NR_STACK_POINTER_REG then                      a_reg_alloc(list,NR_R12);                    a_load_const_reg(list,OS_ADDR,LocalSize,NR_R12);                    list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R12));                    a_reg_dealloc(list,NR_R12);                  end                else                  begin                    a_reg_dealloc(list,NR_R12);                    list.concat(taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,LocalSize));                  end;              end;            if firstfloatreg<>RS_NO then              begin                reference_reset(ref,4);                if tg.direction*tarmprocinfo(current_procinfo).floatregstart>=1023 then                  begin                    a_load_const_reg(list,OS_ADDR,-tarmprocinfo(current_procinfo).floatregstart,NR_R12);                    list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_R12,current_procinfo.framepointer,NR_R12));                    ref.base:=NR_R12;                  end                else                  begin                    ref.base:=current_procinfo.framepointer;                    ref.offset:=tarmprocinfo(current_procinfo).floatregstart;                  end;                list.concat(taicpu.op_reg_const_ref(A_SFM,newreg(R_FPUREGISTER,firstfloatreg,R_SUBWHOLE),                  lastfloatreg-firstfloatreg+1,ref));              end;          end;      end;    procedure tthumb2cgarm.g_proc_exit(list : TAsmList;parasize : longint;nostackframe:boolean);      var         ref : treference;         firstfloatreg,lastfloatreg,         r : byte;         shift : byte;         regs : tcpuregisterset;         LocalSize : longint;         stackmisalignment: pint;      begin        if not(nostackframe) then          begin            stackmisalignment:=0;            { restore floating point register }            firstfloatreg:=RS_NO;            lastfloatreg:=RS_NO;            { save floating point registers? }            for r:=RS_F0 to RS_F7 do              if r in rg[R_FPUREGISTER].used_in_proc-paramanager.get_volatile_registers_fpu(pocall_stdcall) then                begin                  if firstfloatreg=RS_NO then                    firstfloatreg:=r;                  lastfloatreg:=r;                  { floating point register space is already included in                    localsize below by calc_stackframe_size                   inc(stackmisalignment,12);                  }                end;            if firstfloatreg<>RS_NO then              begin                reference_reset(ref,4);                if tg.direction*tarmprocinfo(current_procinfo).floatregstart>=1023 then                  begin                    a_load_const_reg(list,OS_ADDR,-tarmprocinfo(current_procinfo).floatregstart,NR_R12);                    list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_R12,current_procinfo.framepointer,NR_R12));                    ref.base:=NR_R12;                  end                else                  begin                    ref.base:=current_procinfo.framepointer;                    ref.offset:=tarmprocinfo(current_procinfo).floatregstart;                  end;                list.concat(taicpu.op_reg_const_ref(A_LFM,newreg(R_FPUREGISTER,firstfloatreg,R_SUBWHOLE),                  lastfloatreg-firstfloatreg+1,ref));              end;            regs:=rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall);            if (pi_do_call in current_procinfo.flags) or (regs<>[]) then              begin                exclude(regs,RS_R14);                include(regs,RS_R15);              end;            if (current_procinfo.framepointer<>NR_STACK_POINTER_REG) then              regs:=regs+[RS_FRAME_POINTER_REG,RS_R15];            for r:=RS_R0 to RS_R15 do              if (r in regs) then                inc(stackmisalignment,4);            stackmisalignment:=stackmisalignment mod current_settings.alignment.localalignmax;            LocalSize:=current_procinfo.calc_stackframe_size;            if (LocalSize<>0) or               ((stackmisalignment<>0) and                ((pi_do_call in current_procinfo.flags) or                 (po_assembler in current_procinfo.procdef.procoptions))) then              begin                localsize:=align(localsize+stackmisalignment,current_settings.alignment.localalignmax)-stackmisalignment;                if not(is_shifter_const(LocalSize,shift)) then                  begin                    a_reg_alloc(list,NR_R12);                    a_load_const_reg(list,OS_ADDR,LocalSize,NR_R12);                    list.concat(taicpu.op_reg_reg(A_ADD,NR_STACK_POINTER_REG,NR_R12));                    a_reg_dealloc(list,NR_R12);                  end                else                  begin                    a_reg_dealloc(list,NR_R12);                    list.concat(taicpu.op_reg_const(A_ADD,NR_STACK_POINTER_REG,LocalSize));                  end;              end;            if regs=[] then              list.concat(taicpu.op_reg_reg(A_MOV,NR_R15,NR_R14))            else              begin                reference_reset(ref,4);                ref.index:=NR_STACK_POINTER_REG;                ref.addressmode:=AM_PREINDEXED;                list.concat(setoppostfix(taicpu.op_ref_regset(A_LDM,ref,R_INTREGISTER,R_SUBWHOLE,regs),PF_FD));              end;          end        else          list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R14));      end;    function tthumb2cgarm.handle_load_store(list:TAsmList;op: tasmop;oppostfix : toppostfix;reg:tregister;ref: treference):treference;      var        tmpreg : tregister;        tmpref : treference;        l : tasmlabel;        so: tshifterop;      begin        tmpreg:=NR_NO;        { Be sure to have a base register }        if (ref.base=NR_NO) then          begin            if ref.shiftmode<>SM_None then              internalerror(2014020706);            ref.base:=ref.index;            ref.index:=NR_NO;          end;        { absolute symbols can't be handled directly, we've to store the symbol reference          in the text segment and access it pc relative          For now, we assume that references where base or index equals to PC are already          relative, all other references are assumed to be absolute and thus they need          to be handled extra.          A proper solution would be to change refoptions to a set and store the information          if the symbol is absolute or relative there.        }        if (assigned(ref.symbol) and            not(is_pc(ref.base)) and            not(is_pc(ref.index))           ) or           { [#xxx] isn't a valid address operand }           ((ref.base=NR_NO) and (ref.index=NR_NO)) or           //(ref.offset<-4095) or           (ref.offset<-255) or           (ref.offset>4095) or           ((oppostfix in [PF_SB,PF_H,PF_SH]) and            ((ref.offset<-255) or             (ref.offset>255)            )           ) or           (((op in [A_LDF,A_STF,A_FLDS,A_FLDD,A_FSTS,A_FSTD]) or (op=A_VSTR) or (op=A_VLDR)) and            ((ref.offset<-1020) or             (ref.offset>1020) or             ((abs(ref.offset) mod 4)<>0) or             { the usual pc relative symbol handling assumes possible offsets of +/- 4095 }             assigned(ref.symbol)            )           ) then          begin            reference_reset(tmpref,4);            { load symbol }            tmpreg:=getintregister(list,OS_INT);            if assigned(ref.symbol) then              begin                current_asmdata.getjumplabel(l);                cg.a_label(current_procinfo.aktlocaldata,l);                tmpref.symboldata:=current_procinfo.aktlocaldata.last;                current_procinfo.aktlocaldata.concat(tai_const.create_sym_offset(ref.symbol,ref.offset));                { load consts entry }                tmpref.symbol:=l;                tmpref.base:=NR_R15;                list.concat(taicpu.op_reg_ref(A_LDR,tmpreg,tmpref));                { in case of LDF/STF, we got rid of the NR_R15 }                if is_pc(ref.base) then                  ref.base:=NR_NO;                if is_pc(ref.index) then                  ref.index:=NR_NO;              end            else              a_load_const_reg(list,OS_ADDR,ref.offset,tmpreg);            if (ref.base<>NR_NO) then              begin                if ref.index<>NR_NO then                  begin                    list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,tmpreg));                    ref.base:=tmpreg;                  end                else                  begin                    ref.index:=tmpreg;                    ref.shiftimm:=0;                    ref.signindex:=1;                    ref.shiftmode:=SM_None;                  end;              end            else              ref.base:=tmpreg;            ref.offset:=0;            ref.symbol:=nil;          end;        if (ref.base<>NR_NO) and (ref.index<>NR_NO) and (ref.offset<>0) then          begin            if tmpreg<>NR_NO then              a_op_const_reg_reg(list,OP_ADD,OS_ADDR,ref.offset,tmpreg,tmpreg)            else              begin                tmpreg:=getintregister(list,OS_ADDR);                a_op_const_reg_reg(list,OP_ADD,OS_ADDR,ref.offset,ref.base,tmpreg);                ref.base:=tmpreg;              end;            ref.offset:=0;          end;        { Hack? Thumb2 doesn't allow PC indexed addressing modes(although it does in the specification) }        if (ref.base=NR_R15) and (ref.index<>NR_NO) and (ref.shiftmode <> sm_none) then          begin            tmpreg:=getintregister(list,OS_ADDR);            list.concat(taicpu.op_reg_reg(A_MOV, tmpreg, NR_R15));            ref.base := tmpreg;          end;        { floating point operations have only limited references          we expect here, that a base is already set }        if ((op in [A_LDF,A_STF,A_FLDS,A_FLDD,A_FSTS,A_FSTD]) or (op=A_VSTR) or (op=A_VLDR)) and (ref.index<>NR_NO) then          begin            if ref.shiftmode<>SM_none then              internalerror(200309121);            if tmpreg<>NR_NO then              begin                if ref.base=tmpreg then                  begin                    if ref.signindex<0 then                      list.concat(taicpu.op_reg_reg_reg(A_SUB,tmpreg,tmpreg,ref.index))                    else                      list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,tmpreg,ref.index));                    ref.index:=NR_NO;                  end                else                  begin                    if ref.index<>tmpreg then                      internalerror(200403161);                    if ref.signindex<0 then                      list.concat(taicpu.op_reg_reg_reg(A_SUB,tmpreg,ref.base,tmpreg))                    else                      list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,tmpreg));                    ref.base:=tmpreg;                    ref.index:=NR_NO;                  end;              end            else              begin                tmpreg:=getintregister(list,OS_ADDR);                list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,ref.index));                ref.base:=tmpreg;                ref.index:=NR_NO;              end;          end;        list.concat(setoppostfix(taicpu.op_reg_ref(op,reg,ref),oppostfix));        Result := ref;      end;    procedure tthumb2cgarm.a_loadmm_reg_reg(list: TAsmList; fromsize, tosize: tcgsize; reg1, reg2: tregister; shuffle: pmmshuffle);      var        instr: taicpu;      begin        if (fromsize=OS_F32) and          (tosize=OS_F32) then          begin            instr:=setoppostfix(taicpu.op_reg_reg(A_VMOV,reg2,reg1), PF_F32);            list.Concat(instr);            add_move_instruction(instr);          end        else if (fromsize=OS_F64) and          (tosize=OS_F64) then          begin            //list.Concat(setoppostfix(taicpu.op_reg_reg(A_VMOV,tregister(longint(reg2)+1),tregister(longint(reg1)+1)), PF_F32));            //list.Concat(setoppostfix(taicpu.op_reg_reg(A_VMOV,reg2,reg1), PF_F32));          end        else if (fromsize=OS_F32) and          (tosize=OS_F64) then          //list.Concat(setoppostfix(taicpu.op_reg_reg(A_VCVT,reg2,reg1), PF_F32))          begin            //list.concat(nil);          end;      end;    procedure tthumb2cgarm.a_loadmm_ref_reg(list: TAsmList; fromsize, tosize: tcgsize; const ref: treference; reg: tregister; shuffle: pmmshuffle);      begin        if fromsize=OS_F32 then          handle_load_store(list,A_VLDR,PF_F32,reg,ref)        else          handle_load_store(list,A_VLDR,PF_F64,reg,ref);      end;    procedure tthumb2cgarm.a_loadmm_reg_ref(list: TAsmList; fromsize, tosize: tcgsize; reg: tregister; const ref: treference; shuffle: pmmshuffle);      begin        if fromsize=OS_F32 then          handle_load_store(list,A_VSTR,PF_F32,reg,ref)        else          handle_load_store(list,A_VSTR,PF_F64,reg,ref);      end;    procedure tthumb2cgarm.a_loadmm_intreg_reg(list: TAsmList; fromsize, tosize: tcgsize; intreg, mmreg: tregister; shuffle: pmmshuffle);      begin        if //(shuffle=nil) and          (tosize=OS_F32) then          list.Concat(taicpu.op_reg_reg(A_VMOV,mmreg,intreg))        else          internalerror(2012100813);      end;    procedure tthumb2cgarm.a_loadmm_reg_intreg(list: TAsmList; fromsize, tosize: tcgsize; mmreg, intreg: tregister; shuffle: pmmshuffle);      begin        if //(shuffle=nil) and          (fromsize=OS_F32) then          list.Concat(taicpu.op_reg_reg(A_VMOV,intreg,mmreg))        else          internalerror(2012100814);      end;    procedure tthumb2cg64farm.a_op64_reg_reg(list : TAsmList;op:TOpCG;size : tcgsize;regsrc,regdst : tregister64);      var tmpreg: tregister;      begin        case op of          OP_NEG:            begin              cg.a_reg_alloc(list,NR_DEFAULTFLAGS);              list.concat(setoppostfix(taicpu.op_reg_reg_const(A_RSB,regdst.reglo,regsrc.reglo,0),PF_S));              tmpreg:=cg.getintregister(list,OS_32);              list.concat(taicpu.op_reg_const(A_MOV,tmpreg,0));              list.concat(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,tmpreg,regsrc.reghi));              cg.a_reg_dealloc(list,NR_DEFAULTFLAGS);            end;          else            inherited a_op64_reg_reg(list, op, size, regsrc, regdst);        end;      end;    procedure tthumbcg64farm.a_op64_reg_reg(list: TAsmList; op: TOpCG; size: tcgsize; regsrc, regdst: tregister64);      begin        case op of          OP_NEG:            begin              list.concat(taicpu.op_reg_const(A_MOV,regdst.reglo,0));              list.concat(taicpu.op_reg_const(A_MOV,regdst.reghi,0));              cg.a_reg_alloc(list,NR_DEFAULTFLAGS);              list.concat(taicpu.op_reg_reg(A_SUB,regdst.reglo,regsrc.reglo));              list.concat(taicpu.op_reg_reg(A_SBC,regdst.reghi,regsrc.reghi));              cg.a_reg_dealloc(list,NR_DEFAULTFLAGS);            end;          OP_NOT:            begin              cg.a_op_reg_reg(list,OP_NOT,OS_INT,regsrc.reglo,regdst.reglo);              cg.a_op_reg_reg(list,OP_NOT,OS_INT,regsrc.reghi,regdst.reghi);            end;          OP_AND,OP_OR,OP_XOR:            begin              cg.a_op_reg_reg(list,op,OS_32,regsrc.reglo,regdst.reglo);              cg.a_op_reg_reg(list,op,OS_32,regsrc.reghi,regdst.reghi);            end;          OP_ADD:            begin              cg.a_reg_alloc(list,NR_DEFAULTFLAGS);              list.concat(taicpu.op_reg_reg(A_ADD,regdst.reglo,regsrc.reglo));              list.concat(taicpu.op_reg_reg(A_ADC,regdst.reghi,regsrc.reghi));            end;          OP_SUB:            begin              cg.a_reg_alloc(list,NR_DEFAULTFLAGS);              list.concat(taicpu.op_reg_reg(A_SUB,regdst.reglo,regsrc.reglo));              list.concat(taicpu.op_reg_reg(A_SBC,regdst.reghi,regsrc.reghi));            end;          else            internalerror(2003083101);        end;      end;    procedure tthumbcg64farm.a_op64_const_reg(list: TAsmList; op: TOpCG; size: tcgsize; value: int64; reg: tregister64);      var        tmpreg : tregister;        b : byte;      begin        case op of          OP_AND,OP_OR,OP_XOR:            begin              cg.a_op_const_reg(list,op,OS_32,aint(lo(value)),reg.reglo);              cg.a_op_const_reg(list,op,OS_32,aint(hi(value)),reg.reghi);            end;          OP_ADD:            begin              if (aint(lo(value))>=0) and (aint(lo(value))<=255) then                begin                  cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(taicpu.op_reg_const(A_ADD,reg.reglo,aint(lo(value))));                end              else                begin                  tmpreg:=cg.getintregister(list,OS_32);                  cg.a_load_const_reg(list,OS_32,aint(lo(value)),tmpreg);                  cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(taicpu.op_reg_reg(A_ADD,reg.reglo,tmpreg));                end;               tmpreg:=cg.getintregister(list,OS_32);               cg.a_load_const_reg(list,OS_32,aint(hi(value)),tmpreg);               list.concat(taicpu.op_reg_reg(A_ADC,reg.reghi,tmpreg));            end;          OP_SUB:            begin              if (aint(lo(value))>=0) and (aint(lo(value))<=255) then                begin                  cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(taicpu.op_reg_const(A_SUB,reg.reglo,aint(lo(value))))                end              else                begin                  tmpreg:=cg.getintregister(list,OS_32);                  cg.a_load_const_reg(list,OS_32,aint(lo(value)),tmpreg);                  cg.a_reg_alloc(list,NR_DEFAULTFLAGS);                  list.concat(taicpu.op_reg_reg(A_SUB,reg.reglo,tmpreg));                end;              tmpreg:=cg.getintregister(list,OS_32);              cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);              list.concat(taicpu.op_reg_reg(A_SBC,reg.reghi,tmpreg));            end;          else            internalerror(2003083101);        end;      end;    procedure create_codegen;      begin        if GenerateThumb2Code then          begin            cg:=tthumb2cgarm.create;            cg64:=tthumb2cg64farm.create;            casmoptimizer:=TCpuThumb2AsmOptimizer;          end        else if GenerateThumbCode then          begin            cg:=tthumbcgarm.create;            cg64:=tthumbcg64farm.create;            // casmoptimizer:=TCpuThumbAsmOptimizer;          end        else          begin            cg:=tarmcgarm.create;            cg64:=tarmcg64farm.create;            casmoptimizer:=TCpuAsmOptimizer;          end;      end;end.
 |