| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168 | {    $Id$    Copyright (c) 1998-2000 by Florian Klaempfl    Helper routines for the i386 code generator    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 cgai386;  interface    uses       cobjects,tree,       cpubase,cpuasm,       symconst,symtable,aasm;{$define TESTGETTEMP to store const that are written into temps for later release PM }    function def_opsize(p1:pdef):topsize;    function def2def_opsize(p1,p2:pdef):topsize;    function def_getreg(p1:pdef):tregister;    function makereg8(r:tregister):tregister;    function makereg16(r:tregister):tregister;    function makereg32(r:tregister):tregister;    procedure emitlab(var l : pasmlabel);    procedure emitjmp(c : tasmcond;var l : pasmlabel);    procedure emit_flag2reg(flag:tresflags;hregister:tregister);    procedure emit_none(i : tasmop;s : topsize);    procedure emit_const(i : tasmop;s : topsize;c : longint);    procedure emit_reg(i : tasmop;s : topsize;reg : tregister);    procedure emit_ref(i : tasmop;s : topsize;ref : preference);    procedure emit_const_reg(i : tasmop;s : topsize;c : longint;reg : tregister);    procedure emit_const_ref(i : tasmop;s : topsize;c : longint;ref : preference);    procedure emit_ref_reg(i : tasmop;s : topsize;ref : preference;reg : tregister);    procedure emit_reg_ref(i : tasmop;s : topsize;reg : tregister;ref : preference);    procedure emit_reg_reg(i : tasmop;s : topsize;reg1,reg2 : tregister);    procedure emit_const_reg_reg(i : tasmop;s : topsize;c : longint;reg1,reg2 : tregister);    procedure emit_reg_reg_reg(i : tasmop;s : topsize;reg1,reg2,reg3 : tregister);    procedure emit_sym(i : tasmop;s : topsize;op : pasmsymbol);    procedure emit_sym_ofs(i : tasmop;s : topsize;op : pasmsymbol;ofs : longint);    procedure emit_sym_ofs_reg(i : tasmop;s : topsize;op : pasmsymbol;ofs:longint;reg : tregister);    procedure emit_sym_ofs_ref(i : tasmop;s : topsize;op : pasmsymbol;ofs:longint;ref : preference);    procedure emitcall(const routine:string);    procedure emit_mov_loc_ref(const t:tlocation;const ref:treference;siz:topsize;freetemp:boolean);    procedure emit_mov_loc_reg(const t:tlocation;reg:tregister);    procedure emit_mov_ref_reg64(r : treference;rl,rh : tregister);    procedure emit_lea_loc_ref(const t:tlocation;const ref:treference;freetemp:boolean);    procedure emit_lea_loc_reg(const t:tlocation;reg:tregister;freetemp:boolean);    procedure emit_push_loc(const t:tlocation);    { pushes qword location to the stack }    procedure emit_pushq_loc(const t : tlocation);    procedure release_qword_loc(const t : tlocation);    { remove non regvar registers in loc from regs (in the format }    { pushusedregisters uses)                                     }    procedure remove_non_regvars_from_loc(const t: tlocation; var regs: byte);    { releases the registers of a location }    procedure release_loc(const t : tlocation);    procedure emit_pushw_loc(const t:tlocation);    procedure emit_push_lea_loc(const t:tlocation;freetemp:boolean);    procedure emit_to_mem(var p:ptree);    procedure emit_to_reg16(var hr:tregister);    procedure emit_to_reg32(var hr:tregister);    procedure emit_mov_reg_loc(reg: TRegister; const t:tlocation);    procedure emit_movq_reg_loc(reghigh,reglow: TRegister;t:tlocation);    procedure copyshortstring(const dref,sref : treference;len : byte;                        loadref, del_sref: boolean);    procedure loadansistring(p : ptree);    procedure finalize(t : pdef;const ref : treference;is_already_ref : boolean);    procedure incrstringref(t : pdef;const ref : treference);    procedure decrstringref(t : pdef;const ref : treference);    function maybe_push(needed : byte;p : ptree;isint64 : boolean) : boolean;    procedure push_int(l : longint);    procedure emit_push_mem(const ref : treference);    procedure emitpushreferenceaddr(const ref : treference);    procedure pushsetelement(p : ptree);    procedure restore(p : ptree;isint64 : boolean);    procedure push_value_para(p:ptree;inlined,is_cdecl:boolean;                              para_offset:longint;alignment : longint);{$ifdef TEMPS_NOT_PUSH}    { does the same as restore/, but uses temp. space instead of pushing }    function maybe_push(needed : byte;p : ptree;isint64 : boolean) : boolean;    procedure restorefromtemp(p : ptree;isint64 : boolean);{$endif TEMPS_NOT_PUSH}    procedure floatload(t : tfloattype;const ref : treference);    procedure floatstore(t : tfloattype;const ref : treference);    procedure floatloadops(t : tfloattype;var op : tasmop;var s : topsize);    procedure floatstoreops(t : tfloattype;var op : tasmop;var s : topsize);    procedure maybe_loadesi;    procedure maketojumpbool(p : ptree);    procedure emitloadord2reg(const location:Tlocation;orddef:Porddef;destreg:Tregister;delloc:boolean);    procedure emitoverflowcheck(p:ptree);    procedure emitrangecheck(p:ptree;todef:pdef);    procedure concatcopy(source,dest : treference;size : longint;delsource : boolean;loadref:boolean);    procedure firstcomplex(p : ptree);    procedure genentrycode(alist : paasmoutput;const proc_names:Tstringcontainer;make_global:boolean;                           stackframe:longint;                           var parasize:longint;var nostackframe:boolean;                           inlined : boolean);    procedure genexitcode(alist : paasmoutput;parasize:longint;                          nostackframe,inlined:boolean);    { if a unit doesn't have a explicit init/final code,  }    { we've to generate one, if the units has ansistrings }    { in the interface or implementation                  }    procedure genimplicitunitfinal(alist : paasmoutput);    procedure genimplicitunitinit(alist : paasmoutput);{$ifdef test_dest_loc}const  { used to avoid temporary assignments }  dest_loc_known : boolean = false;  in_dest_loc    : boolean = false;  dest_loc_tree  : ptree = nil;var  dest_loc : tlocation;procedure mov_reg_to_dest(p : ptree; s : topsize; reg : tregister);{$endif test_dest_loc}  implementation    uses       strings,globtype,systems,globals,verbose,files,types,pbase,       tgeni386,temp_gen,hcodegen,ppu{$ifdef GDB}       ,gdb{$endif}{$ifndef NOTARGETWIN32}       ,t_win32{$endif}       ;{*****************************************************************************                                Helpers*****************************************************************************}    function def_opsize(p1:pdef):topsize;      begin        case p1^.size of         1 : def_opsize:=S_B;         2 : def_opsize:=S_W;         4 : def_opsize:=S_L;        else         internalerror(78);        end;      end;    function def2def_opsize(p1,p2:pdef):topsize;      var        o1 : topsize;      begin        case p1^.size of         1 : o1:=S_B;         2 : o1:=S_W;         4 : o1:=S_L;         { I don't know if we need it (FK) }         8 : o1:=S_L;        else         internalerror(78);        end;        if assigned(p2) then         begin           case p2^.size of            1 : o1:=S_B;            2 : begin                  if o1=S_B then                   o1:=S_BW                  else                   o1:=S_W;                end;            4,8:              begin                 case o1 of                    S_B : o1:=S_BL;                    S_W : o1:=S_WL;                 end;              end;           end;         end;        def2def_opsize:=o1;      end;    function def_getreg(p1:pdef):tregister;      begin        case p1^.size of         1 : def_getreg:=reg32toreg8(getregister32);         2 : def_getreg:=reg32toreg16(getregister32);         4 : def_getreg:=getregister32;        else         internalerror(78);        end;      end;    function makereg8(r:tregister):tregister;      begin        case r of          R_EAX,R_EBX,R_ECX,R_EDX,R_EDI,R_ESI,R_ESP :            makereg8:=reg32toreg8(r);          R_AX,R_BX,R_CX,R_DX,R_DI,R_SI,R_SP :            makereg8:=reg16toreg8(r);          R_AL,R_BL,R_CL,R_DL :            makereg8:=r;        end;      end;    function makereg16(r:tregister):tregister;      begin        case r of          R_EAX,R_EBX,R_ECX,R_EDX,R_EDI,R_ESI,R_ESP :            makereg16:=reg32toreg16(r);          R_AX,R_BX,R_CX,R_DX,R_DI,R_SI,R_SP :            makereg16:=r;          R_AL,R_BL,R_CL,R_DL :            makereg16:=reg8toreg16(r);        end;      end;    function makereg32(r:tregister):tregister;      begin        case r of          R_EAX,R_EBX,R_ECX,R_EDX,R_EDI,R_ESI,R_ESP :            makereg32:=r;          R_AX,R_BX,R_CX,R_DX,R_DI,R_SI,R_SP :            makereg32:=reg16toreg32(r);          R_AL,R_BL,R_CL,R_DL :            makereg32:=reg8toreg32(r);        end;      end;{*****************************************************************************                              Emit Assembler*****************************************************************************}    procedure emitlab(var l : pasmlabel);      begin         if not l^.is_set then          exprasmlist^.concat(new(pai_label,init(l)))         else          internalerror(7453984);      end;{$ifdef nojmpfix}    procedure emitjmp(c : tasmcond;var l : pasmlabel);      var        ai : Paicpu;      begin        if c=C_None then          exprasmlist^.concat(new(paicpu,op_sym(A_JMP,S_NO,l)))        else          begin            ai:=new(paicpu,op_sym(A_Jcc,S_NO,l));            ai^.SetCondition(c);            ai^.is_jmp:=true;            exprasmlist^.concat(ai);          end;      end;{$else nojmpfix}    procedure emitjmp(c : tasmcond;var l : pasmlabel);      var        ai : Paicpu;      begin        if c=C_None then          ai := new(paicpu,op_sym(A_JMP,S_NO,l))        else          begin            ai:=new(paicpu,op_sym(A_Jcc,S_NO,l));            ai^.SetCondition(c);          end;        ai^.is_jmp:=true;        exprasmlist^.concat(ai);      end;{$endif nojmpfix}    procedure emit_flag2reg(flag:tresflags;hregister:tregister);      var        ai : paicpu;        hreg : tregister;      begin         hreg:=makereg8(hregister);         ai:=new(paicpu,op_reg(A_Setcc,S_B,hreg));         ai^.SetCondition(flag_2_cond[flag]);         exprasmlist^.concat(ai);         if hreg<>hregister then          begin            if hregister in regset16bit then             emit_to_reg16(hreg)            else             emit_to_reg32(hreg);          end;      end;    procedure emit_none(i : tasmop;s : topsize);      begin         exprasmlist^.concat(new(paicpu,op_none(i,s)));      end;    procedure emit_reg(i : tasmop;s : topsize;reg : tregister);      begin         exprasmlist^.concat(new(paicpu,op_reg(i,s,reg)));      end;    procedure emit_ref(i : tasmop;s : topsize;ref : preference);      begin         exprasmlist^.concat(new(paicpu,op_ref(i,s,ref)));      end;    procedure emit_const(i : tasmop;s : topsize;c : longint);      begin         exprasmlist^.concat(new(paicpu,op_const(i,s,c)));      end;    procedure emit_const_reg(i : tasmop;s : topsize;c : longint;reg : tregister);      begin         exprasmlist^.concat(new(paicpu,op_const_reg(i,s,c,reg)));      end;    procedure emit_const_ref(i : tasmop;s : topsize;c : longint;ref : preference);      begin         exprasmlist^.concat(new(paicpu,op_const_ref(i,s,c,ref)));      end;    procedure emit_ref_reg(i : tasmop;s : topsize;ref : preference;reg : tregister);      begin         exprasmlist^.concat(new(paicpu,op_ref_reg(i,s,ref,reg)));      end;    procedure emit_reg_ref(i : tasmop;s : topsize;reg : tregister;ref : preference);      begin         exprasmlist^.concat(new(paicpu,op_reg_ref(i,s,reg,ref)));      end;    procedure emit_reg_reg(i : tasmop;s : topsize;reg1,reg2 : tregister);      begin         if (reg1<>reg2) or (i<>A_MOV) then           exprasmlist^.concat(new(paicpu,op_reg_reg(i,s,reg1,reg2)));      end;    procedure emit_const_reg_reg(i : tasmop;s : topsize;c : longint;reg1,reg2 : tregister);      begin         exprasmlist^.concat(new(paicpu,op_const_reg_reg(i,s,c,reg1,reg2)));      end;    procedure emit_reg_reg_reg(i : tasmop;s : topsize;reg1,reg2,reg3 : tregister);      begin         exprasmlist^.concat(new(paicpu,op_reg_reg_reg(i,s,reg1,reg2,reg3)));      end;    procedure emit_sym(i : tasmop;s : topsize;op : pasmsymbol);      begin        exprasmlist^.concat(new(paicpu,op_sym(i,s,op)));      end;    procedure emit_sym_ofs(i : tasmop;s : topsize;op : pasmsymbol;ofs : longint);      begin        exprasmlist^.concat(new(paicpu,op_sym_ofs(i,s,op,ofs)));      end;    procedure emit_sym_ofs_reg(i : tasmop;s : topsize;op : pasmsymbol;ofs:longint;reg : tregister);      begin        exprasmlist^.concat(new(paicpu,op_sym_ofs_reg(i,s,op,ofs,reg)));      end;    procedure emit_sym_ofs_ref(i : tasmop;s : topsize;op : pasmsymbol;ofs:longint;ref : preference);      begin        exprasmlist^.concat(new(paicpu,op_sym_ofs_ref(i,s,op,ofs,ref)));      end;    procedure emitcall(const routine:string);      begin        exprasmlist^.concat(new(paicpu,op_sym(A_CALL,S_NO,newasmsymbol(routine))));      end;    { only usefull in startup code }    procedure emitinsertcall(const routine:string);      begin        exprasmlist^.insert(new(paicpu,op_sym(A_CALL,S_NO,newasmsymbol(routine))));      end;    procedure emit_mov_loc_ref(const t:tlocation;const ref:treference;siz:topsize;freetemp:boolean);      var        hreg : tregister;        pushedeax : boolean;      begin        pushedeax:=false;        case t.loc of          LOC_REGISTER,         LOC_CREGISTER : begin                           exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,siz,                             t.register,newreference(ref))));                           ungetregister32(t.register); { the register is not needed anymore }                         end;               LOC_MEM,         LOC_REFERENCE : begin                           if t.reference.is_immediate then                             emit_const_ref(A_MOV,siz,                               t.reference.offset,newreference(ref))                           else                             begin                               case siz of                                 S_B : begin                                          { we can't do a getregister in the code generator }                                          { without problems!!!                             }                                          if usablereg32>0 then                                            hreg:=reg32toreg8(getregister32)                                          else                                            begin                                               emit_reg(A_PUSH,S_L,R_EAX);                                               pushedeax:=true;                                               hreg:=R_AL;                                            end;                                       end;                                 S_W : hreg:=R_DI;                                 S_L : hreg:=R_EDI;                               end;{$ifndef noAllocEdi}                               if hreg in [R_DI,R_EDI] then                                 getexplicitregister32(R_EDI);{$endif noAllocEdi}                               emit_ref_reg(A_MOV,siz,                                 newreference(t.reference),hreg);                               del_reference(t.reference);                               exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,siz,                                 hreg,newreference(ref))));                               if siz=S_B then                                 begin                                    if pushedeax then                                      emit_reg(A_POP,S_L,R_EAX)                                    else                                      ungetregister(hreg);                                 end;{$ifndef noAllocEdi}                               if hreg in [R_DI,R_EDI] then                                 ungetregister32(R_EDI);{$endif noAllocEdi}                               { we can release the registers }                               { but only AFTER the MOV! Important for the optimizer!                                 (JM)}                               del_reference(ref);                             end;                           if freetemp then                            ungetiftemp(t.reference);                         end;        else         internalerror(330);        end;      end;    procedure emit_mov_loc_reg(const t:tlocation;reg:tregister);      begin        case t.loc of          LOC_REGISTER,         LOC_CREGISTER : begin                           emit_reg_reg(A_MOV,S_L,t.register,reg);                           ungetregister32(t.register); { the register is not needed anymore }                         end;               LOC_MEM,         LOC_REFERENCE : begin                           if t.reference.is_immediate then                             emit_const_reg(A_MOV,S_L,                               t.reference.offset,reg)                           else                             begin                               emit_ref_reg(A_MOV,S_L,                                 newreference(t.reference),reg);                             end;                         end;        else         internalerror(330);        end;      end;    procedure emit_mov_reg_loc(reg: TRegister; const t:tlocation);      begin        case t.loc of          LOC_REGISTER,         LOC_CREGISTER : begin                           emit_reg_reg(A_MOV,RegSize(Reg),                             reg,t.register);                         end;               LOC_MEM,         LOC_REFERENCE : begin                           if t.reference.is_immediate then                             internalerror(334)                           else                             begin                               exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,RegSize(Reg),                                 Reg,newreference(t.reference))));                             end;                         end;        else         internalerror(330);        end;      end;    procedure emit_lea_loc_reg(const t:tlocation;reg:tregister;freetemp:boolean);      begin        case t.loc of               LOC_MEM,         LOC_REFERENCE : begin                           if t.reference.is_immediate then                             internalerror(331)                           else                             begin                               emit_ref_reg(A_LEA,S_L,                                 newreference(t.reference),reg);                             end;                           if freetemp then                            ungetiftemp(t.reference);                         end;        else         internalerror(332);        end;      end;    procedure emit_movq_reg_loc(reghigh,reglow: TRegister;t:tlocation);      begin        case t.loc of          LOC_REGISTER,         LOC_CREGISTER : begin                           emit_reg_reg(A_MOV,S_L,                             reglow,t.registerlow);                           emit_reg_reg(A_MOV,S_L,                             reghigh,t.registerhigh);                         end;               LOC_MEM,         LOC_REFERENCE : begin                           if t.reference.is_immediate then                             internalerror(334)                           else                             begin                               exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,                                 Reglow,newreference(t.reference))));                               inc(t.reference.offset,4);                               exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,                                 Reghigh,newreference(t.reference))));                             end;                         end;        else         internalerror(330);        end;      end;   procedure emit_pushq_loc(const t : tlocation);      var         hr : preference;      begin         case t.loc of            LOC_REGISTER,            LOC_CREGISTER:              begin                 exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,                   t.registerhigh)));                 exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,                   t.registerlow)));              end;            LOC_MEM,            LOC_REFERENCE:              begin                 hr:=newreference(t.reference);                 inc(hr^.offset,4);                 exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,S_L,                   hr)));                 exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,S_L,                   newreference(t.reference))));                 ungetiftemp(t.reference);              end;            else internalerror(331);         end;      end;    procedure remove_non_regvars_from_loc(const t: tlocation; var regs: byte);    begin      case t.loc of        LOC_REGISTER:          { can't be a regvar, since it would be LOC_CREGISTER then }          regs := regs and not($80 shr byte(t.register));        LOC_MEM,LOC_REFERENCE:          begin            if not(cs_regalloc in aktglobalswitches) or               (t.reference.base in usableregs) then              regs := regs and                not($80 shr byte(t.reference.base));            if not(cs_regalloc in aktglobalswitches) or               (t.reference.index in usableregs) then              regs := regs and                not($80 shr byte(t.reference.index));          end;      end;    end;    procedure release_loc(const t : tlocation);      begin         case t.loc of            LOC_REGISTER,            LOC_CREGISTER:              begin                 ungetregister32(t.register);              end;            LOC_MEM,            LOC_REFERENCE:              del_reference(t.reference);            else internalerror(332);         end;      end;    procedure release_qword_loc(const t : tlocation);      begin         case t.loc of            LOC_REGISTER,            LOC_CREGISTER:              begin                 ungetregister32(t.registerhigh);                 ungetregister32(t.registerlow);              end;            LOC_MEM,            LOC_REFERENCE:              del_reference(t.reference);            else internalerror(331);         end;      end;    procedure emit_push_loc(const t:tlocation);      begin        case t.loc of          LOC_REGISTER,         LOC_CREGISTER : begin                           exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,makereg32(t.register))));                           ungetregister(t.register); { the register is not needed anymore }                         end;               LOC_MEM,         LOC_REFERENCE : begin                           if t.reference.is_immediate then                             exprasmlist^.concat(new(paicpu,op_const(A_PUSH,S_L,t.reference.offset)))                           else                             exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,S_L,newreference(t.reference))));                           del_reference(t.reference);                           ungetiftemp(t.reference);                         end;        else         internalerror(330);        end;      end;    procedure emit_pushw_loc(const t:tlocation);      var        opsize : topsize;      begin        case t.loc of          LOC_REGISTER,         LOC_CREGISTER : begin                           if target_os.stackalignment=4 then                             exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,makereg32(t.register))))                           else                             exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_W,makereg16(t.register))));                           ungetregister(t.register); { the register is not needed anymore }                         end;               LOC_MEM,         LOC_REFERENCE : begin                           if target_os.stackalignment=4 then                            opsize:=S_L                           else                            opsize:=S_W;                           if t.reference.is_immediate then                             exprasmlist^.concat(new(paicpu,op_const(A_PUSH,opsize,t.reference.offset)))                           else                             exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,opsize,newreference(t.reference))));                           del_reference(t.reference);                           ungetiftemp(t.reference);                         end;        else         internalerror(330);        end;      end;    procedure emit_lea_loc_ref(const t:tlocation;const ref:treference;freetemp:boolean);      begin        case t.loc of               LOC_MEM,         LOC_REFERENCE : begin                           if t.reference.is_immediate then                             internalerror(331)                           else                             begin{$ifndef noAllocEdi}                               getexplicitregister32(R_EDI);{$endif noAllocEdi}                               emit_ref_reg(A_LEA,S_L,                                 newreference(t.reference),R_EDI);                               exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,                                 R_EDI,newreference(ref))));{$ifndef noAllocEdi}                               ungetregister32(R_EDI);{$endif noAllocEdi}                             end;                            { release the registers }                            del_reference(t.reference);                           if freetemp then                            ungetiftemp(t.reference);                         end;        else         internalerror(332);        end;      end;    procedure emit_push_lea_loc(const t:tlocation;freetemp:boolean);      begin        case t.loc of               LOC_MEM,         LOC_REFERENCE : begin                           if t.reference.is_immediate then                             internalerror(331)                           else                             begin{$ifndef noAllocEdi}                               getexplicitregister32(R_EDI);{$endif noAllocEdi}                               emit_ref_reg(A_LEA,S_L,                                 newreference(t.reference),R_EDI);                               exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,R_EDI)));{$ifndef noAllocEdi}                               ungetregister32(R_EDI);{$endif noAllocEdi}                             end;                           if freetemp then                            ungetiftemp(t.reference);                         end;        else         internalerror(332);        end;      end;    procedure emit_to_mem(var p:ptree);      var         r : treference;      begin        case p^.location.loc of               LOC_FPU : begin                           reset_reference(p^.location.reference);                           gettempofsizereference(10,p^.location.reference);                           floatstore(pfloatdef(p^.resulttype)^.typ,p^.location.reference);                           {  This can't be never a l-value! (FK)                              p^.location.loc:=LOC_REFERENCE; }                         end;               LOC_REGISTER:                 begin                    if is_64bitint(p^.resulttype) then                      begin                         gettempofsizereference(8,r);                         emit_reg_ref(A_MOV,S_L,p^.location.registerlow,                           newreference(r));                         inc(r.offset,4);                         emit_reg_ref(A_MOV,S_L,p^.location.registerhigh,                           newreference(r));                         dec(r.offset,4);                         p^.location.reference:=r;                      end                    else                      internalerror(1405001);                 end;               LOC_MEM,         LOC_REFERENCE : ;         LOC_CFPUREGISTER : begin                           emit_reg(A_FLD,S_NO,correct_fpuregister(p^.location.register,fpuvaroffset));                           inc(fpuvaroffset);                           reset_reference(p^.location.reference);                           gettempofsizereference(10,p^.location.reference);                           floatstore(pfloatdef(p^.resulttype)^.typ,p^.location.reference);                           {  This can't be never a l-value! (FK)                              p^.location.loc:=LOC_REFERENCE; }                         end;         else         internalerror(333);        end;        p^.location.loc:=LOC_MEM;      end;    procedure emit_to_reg16(var hr:tregister);      begin        { ranges are a little bit bug sensitive ! }        case hr of           R_EAX,R_EBX,R_ECX,R_EDX,R_EDI,R_ESI,R_ESP,R_EBP:             begin               hr:=reg32toreg16(hr);             end;           R_AL,R_BL,R_CL,R_DL:             begin               hr:=reg8toreg16(hr);               emit_const_reg(A_AND,S_W,$ff,hr);             end;           R_AH,R_BH,R_CH,R_DH:             begin               hr:=reg8toreg16(hr);               emit_const_reg(A_AND,S_W,$ff00,hr);             end;        end;      end;    procedure emit_to_reg32(var hr:tregister);      begin        { ranges are a little bit bug sensitive ! }        case hr of           R_AX,R_BX,R_CX,R_DX,R_DI,R_SI,R_SP,R_BP:             begin                hr:=reg16toreg32(hr);                emit_const_reg(A_AND,S_L,$ffff,hr);             end;           R_AL,R_BL,R_CL,R_DL:             begin                hr:=reg8toreg32(hr);                emit_const_reg(A_AND,S_L,$ff,hr);             end;           R_AH,R_BH,R_CH,R_DH:             begin                hr:=reg8toreg32(hr);                emit_const_reg(A_AND,S_L,$ff00,hr);             end;        end;      end;    procedure emit_mov_ref_reg64(r : treference;rl,rh : tregister);      var         hr : preference;      begin         { if we load a 64 bit reference, we must be careful because }         { we could overwrite the registers of the reference by      }         { accident                                                  }         if r.base=rl then           begin              emit_reg_reg(A_MOV,S_L,r.base,                R_EDI);              r.base:=R_EDI;           end         else if r.index=rl then           begin              emit_reg_reg(A_MOV,S_L,r.index,                R_EDI);              r.index:=R_EDI;           end;         emit_ref_reg(A_MOV,S_L,           newreference(r),rl);         hr:=newreference(r);         inc(hr^.offset,4);         emit_ref_reg(A_MOV,S_L,           hr,rh);      end;{*****************************************************************************                           Emit String Functions*****************************************************************************}    procedure copyshortstring(const dref,sref : treference;len : byte;                loadref, del_sref: boolean);      begin         emitpushreferenceaddr(dref);          { if it's deleted right before it's used, the optimizer can move }          { the reg deallocations to the right places (JM)                 }         if del_sref then           del_reference(sref);         if loadref then          emit_push_mem(sref)         else          emitpushreferenceaddr(sref);         push_int(len);         emitcall('FPC_SHORTSTR_COPY');         maybe_loadesi;      end;    procedure copylongstring(const dref,sref : treference;len : longint;loadref:boolean);      begin         emitpushreferenceaddr(dref);         if loadref then          emit_push_mem(sref)         else          emitpushreferenceaddr(sref);         push_int(len);         emitcall('FPC_LONGSTR_COPY');         maybe_loadesi;      end;    procedure incrstringref(t : pdef;const ref : treference);      var         pushedregs : tpushed;      begin         pushusedregisters(pushedregs,$ff);         emitpushreferenceaddr(ref);         if is_ansistring(t) then           begin              emitcall('FPC_ANSISTR_INCR_REF');           end         else if is_widestring(t) then           begin              emitcall('FPC_WIDESTR_INCR_REF');           end         else internalerror(1859);         popusedregisters(pushedregs);      end;    procedure decrstringref(t : pdef;const ref : treference);      var         pushedregs : tpushed;      begin         pushusedregisters(pushedregs,$ff);         emitpushreferenceaddr(ref);         if is_ansistring(t) then           begin              emitcall('FPC_ANSISTR_DECR_REF');           end         else if is_widestring(t) then           begin              emitcall('FPC_WIDESTR_DECR_REF');           end         else internalerror(1859);         popusedregisters(pushedregs);      end;    procedure loadansistring(p : ptree);    {      copies an ansistring from p^.right to p^.left, we      assume, that both sides are ansistring, firstassignement have      to take care of that, an ansistring can't be a register variable    }      var         pushed : tpushed;         regs_to_push: byte;         ungettemp : boolean;      begin         { before pushing any parameter, we have to save all used      }         { registers, but before that we have to release the       }         { registers of that node to save uneccessary pushed       }         { so be careful, if you think you can optimize that code (FK) }         { nevertheless, this has to be changed, because otherwise the }         { register is released before it's contents are pushed ->     }         { problems with the optimizer (JM)                            }         del_reference(p^.left^.location.reference);         ungettemp:=false;         { Find out which registers have to be pushed (JM) }         regs_to_push := $ff;         remove_non_regvars_from_loc(p^.right^.location,regs_to_push);         { And push them (JM) }         pushusedregisters(pushed,regs_to_push);         case p^.right^.location.loc of            LOC_REGISTER,LOC_CREGISTER:              begin                 exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,p^.right^.location.register)));                 ungetregister32(p^.right^.location.register);              end;            LOC_REFERENCE,LOC_MEM:              begin                 { First release the registers because emit_push_mem may  }                 { load the reference in edi before pushing and then the  }                 { dealloc is too late (and optimizations are missed (JM) }                 del_reference(p^.right^.location.reference);                 { This one doesn't need extra registers (JM) }                 emit_push_mem(p^.right^.location.reference);                 ungettemp:=true;              end;         end;         emitpushreferenceaddr(p^.left^.location.reference);         del_reference(p^.left^.location.reference);         emitcall('FPC_ANSISTR_ASSIGN');         maybe_loadesi;         popusedregisters(pushed);         if ungettemp then           ungetiftemp(p^.right^.location.reference);      end;{*****************************************************************************                           Emit Push Functions*****************************************************************************}    function maybe_push(needed : byte;p : ptree;isint64 : boolean) : boolean;      var         pushed : boolean;         {hregister : tregister; }{$ifdef TEMPS_NOT_PUSH}         href : treference;{$endif TEMPS_NOT_PUSH}      begin         if needed>usablereg32 then           begin              if (p^.location.loc=LOC_REGISTER) then                begin                   if isint64 then                     begin{$ifdef TEMPS_NOT_PUSH}                        gettempofsizereference(href,8);                        p^.temp_offset:=href.offset;                        href.offset:=href.offset+4;                        exprasmlist^.concat(new(paicpu,op_reg(A_MOV,S_L,p^.location.registerhigh,href)));                        href.offset:=href.offset-4;{$else TEMPS_NOT_PUSH}                        exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,p^.location.registerhigh)));{$endif TEMPS_NOT_PUSH}                        ungetregister32(p^.location.registerhigh);                     end{$ifdef TEMPS_NOT_PUSH}                   else                     begin                        gettempofsizereference(href,4);                        p^.temp_offset:=href.offset;                     end{$endif TEMPS_NOT_PUSH}                     ;                   pushed:=true;{$ifdef TEMPS_NOT_PUSH}                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,p^.location.register,href)));{$else TEMPS_NOT_PUSH}                   exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,p^.location.register)));{$endif TEMPS_NOT_PUSH}                   ungetregister32(p^.location.register);                end              else if (p^.location.loc in [LOC_MEM,LOC_REFERENCE]) and                      ((p^.location.reference.base<>R_NO) or                       (p^.location.reference.index<>R_NO)                      ) then                  begin                     del_reference(p^.location.reference);{$ifndef noAllocEdi}                    getexplicitregister32(R_EDI);{$endif noAllocEdi}                     emit_ref_reg(A_LEA,S_L,newreference(p^.location.reference),                       R_EDI);{$ifdef TEMPS_NOT_PUSH}                     gettempofsizereference(href,4);                     exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,href)));                     p^.temp_offset:=href.offset;{$else TEMPS_NOT_PUSH}                     exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,R_EDI)));{$endif TEMPS_NOT_PUSH}{$ifndef noAllocEdi}                     ungetregister32(R_EDI);{$endif noAllocEdi}                     pushed:=true;                  end              else pushed:=false;           end         else pushed:=false;         maybe_push:=pushed;      end;{$ifdef TEMPS_NOT_PUSH}    function maybe_savetotemp(needed : byte;p : ptree;isint64 : boolean) : boolean;      var         pushed : boolean;         href : treference;      begin         if needed>usablereg32 then           begin              if (p^.location.loc=LOC_REGISTER) then                begin                   if isint64(p^.resulttype) then                     begin                        gettempofsizereference(href,8);                        p^.temp_offset:=href.offset;                        href.offset:=href.offset+4;                        exprasmlist^.concat(new(paicpu,op_reg(A_MOV,S_L,p^.location.registerhigh,href)));                        href.offset:=href.offset-4;                        ungetregister32(p^.location.registerhigh);                     end                   else                     begin                        gettempofsizereference(href,4);                        p^.temp_offset:=href.offset;                     end;                   pushed:=true;                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,p^.location.register,href)));                   ungetregister32(p^.location.register);                end              else if (p^.location.loc in [LOC_MEM,LOC_REFERENCE]) and                      ((p^.location.reference.base<>R_NO) or                       (p^.location.reference.index<>R_NO)                      ) then                  begin                     del_reference(p^.location.reference);{$ifndef noAllocEdi}                     getexplicitregister32(R_EDI);{$endif noAllocEdi}                     emit_ref_reg(A_LEA,S_L,newreference(p^.location.reference),                       R_EDI);                     gettempofsizereference(href,4);                     exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,href)));{$ifndef noAllocEdi}                     ungetregister32(R_EDI);{$endif noAllocEdi}                     p^.temp_offset:=href.offset;                     pushed:=true;                  end              else pushed:=false;           end         else pushed:=false;         maybe_push:=pushed;      end;{$endif TEMPS_NOT_PUSH}    procedure push_int(l : longint);      begin         if (l = 0) and            not(aktoptprocessor in [Class386, ClassP6]) and            not(cs_littlesize in aktglobalswitches)           Then             begin{$ifndef noAllocEdi}               getexplicitregister32(R_EDI);{$endif noAllocEdi}               emit_reg_reg(A_XOR,S_L,R_EDI,R_EDI);               exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,R_EDI)));{$ifndef noAllocEdi}               ungetregister32(R_EDI);{$endif noAllocEdi}             end           else             exprasmlist^.concat(new(paicpu,op_const(A_PUSH,S_L,l)));      end;    procedure emit_push_mem(const ref : treference);      begin         if ref.is_immediate then           push_int(ref.offset)         else           begin             if not(aktoptprocessor in [Class386, ClassP6]) and                not(cs_littlesize in aktglobalswitches)               then                 begin{$ifndef noAllocEdi}                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                   emit_ref_reg(A_MOV,S_L,newreference(ref),R_EDI);                   exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,R_EDI)));{$ifndef noAllocEdi}                   ungetregister32(R_EDI);{$endif noAllocEdi}                 end               else exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,S_L,newreference(ref))));           end;      end;    procedure emitpushreferenceaddr(const ref : treference);      var        href : treference;      begin         { this will fail for references to other segments !!! }         if ref.is_immediate then         { is this right ? }           begin              { push_int(ref.offset)}              gettempofsizereference(4,href);              emit_const_ref(A_MOV,S_L,ref.offset,newreference(href));              emitpushreferenceaddr(href);              del_reference(href);           end         else           begin              if ref.segment<>R_NO then                CGMessage(cg_e_cant_use_far_pointer_there);              if (ref.base=R_NO) and (ref.index=R_NO) then                exprasmlist^.concat(new(paicpu,op_sym_ofs(A_PUSH,S_L,ref.symbol,ref.offset)))              else if (ref.base=R_NO) and (ref.index<>R_NO) and                 (ref.offset=0) and (ref.scalefactor=0) and (ref.symbol=nil) then                exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,ref.index)))              else if (ref.base<>R_NO) and (ref.index=R_NO) and                 (ref.offset=0) and (ref.symbol=nil) then                exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,ref.base)))              else                begin{$ifndef noAllocEdi}                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                   emit_ref_reg(A_LEA,S_L,newreference(ref),R_EDI);                   exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,R_EDI)));{$ifndef noAllocEdi}                   ungetregister32(R_EDI);{$endif noAllocEdi}                end;           end;        end;     procedure pushsetelement(p : ptree);     {       copies p a set element on the stack     }      var         hr,hr16,hr32 : tregister;      begin      { copy the element on the stack, slightly complicated }        if p^.treetype=ordconstn then         begin           if target_os.stackalignment=4 then             exprasmlist^.concat(new(paicpu,op_const(A_PUSH,S_L,p^.value)))           else             exprasmlist^.concat(new(paicpu,op_const(A_PUSH,S_W,p^.value)));         end        else         begin           case p^.location.loc of             LOC_REGISTER,             LOC_CREGISTER :               begin                 hr:=p^.location.register;                 case hr of                   R_EAX,R_EBX,R_ECX,R_EDX,R_EDI,R_ESI,R_ESP :                     begin                       hr16:=reg32toreg16(hr);                       hr32:=hr;                     end;                   R_AX,R_BX,R_CX,R_DX,R_DI,R_SI,R_SP :                     begin                       hr16:=hr;                       hr32:=reg16toreg32(hr);                     end;                   R_AL,R_BL,R_CL,R_DL :                     begin                       hr16:=reg8toreg16(hr);                       hr32:=reg8toreg32(hr);                     end;                 end;                 if target_os.stackalignment=4 then                   exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,hr32)))                 else                   exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_W,hr16)));                 ungetregister32(hr32);               end;           else             begin               if target_os.stackalignment=4 then                 exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,S_L,newreference(p^.location.reference))))               else                 exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,S_W,newreference(p^.location.reference))));               del_reference(p^.location.reference);             end;           end;         end;      end;    procedure restore(p : ptree;isint64 : boolean);      var         hregister :  tregister;{$ifdef TEMPS_NOT_PUSH}         href : treference;{$endif TEMPS_NOT_PUSH}      begin         hregister:=getregister32;{$ifdef TEMPS_NOT_PUSH}         reset_reference(href);         href.base:=procinfo^.frame_pointer;         href.offset:=p^.temp_offset;         emit_ref_reg(A_MOV,S_L,href,hregister);{$else  TEMPS_NOT_PUSH}         exprasmlist^.concat(new(paicpu,op_reg(A_POP,S_L,hregister)));{$endif TEMPS_NOT_PUSH}         if (p^.location.loc in [LOC_REGISTER,LOC_CREGISTER]) then           begin              p^.location.register:=hregister;              if isint64 then                begin                   p^.location.registerhigh:=getregister32;{$ifdef TEMPS_NOT_PUSH}                   href.offset:=p^.temp_offset+4;                   emit_ref_reg(A_MOV,S_L,p^.location.registerhigh);                   { set correctly for release ! }                   href.offset:=p^.temp_offset;{$else  TEMPS_NOT_PUSH}                   exprasmlist^.concat(new(paicpu,op_reg(A_POP,S_L,p^.location.registerhigh)));{$endif TEMPS_NOT_PUSH}                end;           end         else           begin              reset_reference(p^.location.reference);              { any reasons why this was moved into the index register ? }              { normally usage of base register is much better (FK)      }              p^.location.reference.base:=hregister;              { Why is this done? We can never be sure about p^.left                because otherwise secondload fails !!!              set_location(p^.left^.location,p^.location);}           end;{$ifdef TEMPS_NOT_PUSH}         ungetiftemp(href);{$endif TEMPS_NOT_PUSH}      end;{$ifdef TEMPS_NOT_PUSH}    procedure restorefromtemp(p : ptree;isint64 : boolean);      var         hregister :  tregister;         href : treference;      begin         hregister:=getregister32;         reset_reference(href);         href.base:=procinfo^.frame_pointer;         href.offset:=p^.temp_offset;         emit_ref_reg(A_MOV,S_L,href,hregister);         if (p^.location.loc in [LOC_REGISTER,LOC_CREGISTER]) then           begin              p^.location.register:=hregister;              if isint64 then                begin                   p^.location.registerhigh:=getregister32;                   href.offset:=p^.temp_offset+4;                   emit_ref_reg(A_MOV,S_L,p^.location.registerhigh);                   { set correctly for release ! }                   href.offset:=p^.temp_offset;                end;           end         else           begin              reset_reference(p^.location.reference);              p^.location.reference.base:=hregister;              { Why is this done? We can never be sure about p^.left                because otherwise secondload fails PM              set_location(p^.left^.location,p^.location);}           end;         ungetiftemp(href);      end;{$endif TEMPS_NOT_PUSH}      procedure push_value_para(p:ptree;inlined,is_cdecl:boolean;                                para_offset:longint;alignment : longint);        var          tempreference : treference;          r : preference;          opsize : topsize;          op : tasmop;          hreg : tregister;          size : longint;          hlabel : pasmlabel;        begin          case p^.location.loc of             LOC_REGISTER,             LOC_CREGISTER:               begin                  case p^.location.register of                     R_EAX,R_EBX,R_ECX,R_EDX,R_ESI,                     R_EDI,R_ESP,R_EBP :                        begin                          if p^.resulttype^.size=8 then                            begin                               inc(pushedparasize,8);                               if inlined then                                 begin                                    r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                    exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,                                      p^.location.registerlow,r)));                                    r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize+4);                                    exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,                                      p^.location.registerhigh,r)));                                 end                               else                                 exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,p^.location.registerhigh)));                               ungetregister32(p^.location.registerhigh);                                 exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,p^.location.registerlow)));                               ungetregister32(p^.location.registerlow);                            end                          else                            begin                               inc(pushedparasize,4);                               if inlined then                                 begin                                    r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                    exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,                                      p^.location.register,r)));                                 end                               else                                 exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,p^.location.register)));                               ungetregister32(p^.location.register);                            end;                        end;                     R_AX,R_BX,R_CX,R_DX,R_SI,R_DI:                        begin                          if alignment=4 then                            begin                              opsize:=S_L;                              hreg:=reg16toreg32(p^.location.register);                              inc(pushedparasize,4);                            end                          else                            begin                              opsize:=S_W;                              hreg:=p^.location.register;                              inc(pushedparasize,2);                            end;                          if inlined then                            begin                              r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                              exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,opsize,hreg,r)));                            end                          else                            exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,opsize,hreg)));                          ungetregister32(reg16toreg32(p^.location.register));                        end;                     R_AL,R_BL,R_CL,R_DL:                        begin                          if alignment=4 then                            begin                              opsize:=S_L;                              hreg:=reg8toreg32(p^.location.register);                              inc(pushedparasize,4);                            end                          else                            begin                              opsize:=S_W;                              hreg:=reg8toreg16(p^.location.register);                              inc(pushedparasize,2);                            end;                          { we must push always 16 bit }                          if inlined then                            begin                              r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                              exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,opsize,hreg,r)));                            end                          else                            exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,opsize,hreg)));                          ungetregister32(reg8toreg32(p^.location.register));                        end;                     else internalerror(1899);                  end;               end;             LOC_FPU:               begin                  size:=align(pfloatdef(p^.resulttype)^.size,alignment);                  inc(pushedparasize,size);                  if not inlined then                   emit_const_reg(A_SUB,S_L,size,R_ESP);{$ifdef GDB}                  if (cs_debuginfo in aktmoduleswitches) and                     (exprasmlist^.first=exprasmlist^.last) then                    exprasmlist^.concat(new(pai_force_line,init));{$endif GDB}                  r:=new_reference(R_ESP,0);                  floatstoreops(pfloatdef(p^.resulttype)^.typ,op,opsize);                  { this is the easiest case for inlined !! }                  if inlined then                    begin                       r^.base:=procinfo^.framepointer;                       r^.offset:=para_offset-pushedparasize;                    end;                  exprasmlist^.concat(new(paicpu,op_ref(op,opsize,r)));                  dec(fpuvaroffset);               end;             LOC_CFPUREGISTER:               begin                  exprasmlist^.concat(new(paicpu,op_reg(A_FLD,S_NO,                    correct_fpuregister(p^.location.register,fpuvaroffset))));                  size:=align(pfloatdef(p^.resulttype)^.size,alignment);                  inc(pushedparasize,size);                  if not inlined then                   emit_const_reg(A_SUB,S_L,size,R_ESP);{$ifdef GDB}                  if (cs_debuginfo in aktmoduleswitches) and                     (exprasmlist^.first=exprasmlist^.last) then                    exprasmlist^.concat(new(pai_force_line,init));{$endif GDB}                  r:=new_reference(R_ESP,0);                  floatstoreops(pfloatdef(p^.resulttype)^.typ,op,opsize);                  { this is the easiest case for inlined !! }                  if inlined then                    begin                       r^.base:=procinfo^.framepointer;                       r^.offset:=para_offset-pushedparasize;                    end;                  exprasmlist^.concat(new(paicpu,op_ref(op,opsize,r)));               end;             LOC_REFERENCE,LOC_MEM:               begin                  tempreference:=p^.location.reference;                  del_reference(p^.location.reference);                  case p^.resulttype^.deftype of                    enumdef,                    orddef :                      begin                        case p^.resulttype^.size of                         8 : begin                               inc(pushedparasize,8);                               if inlined then                                 begin{$ifndef noAllocEdi}                                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                                   emit_ref_reg(A_MOV,S_L,                                     newreference(tempreference),R_EDI);                                   r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,r)));{$ifndef noAllocEdi}                                   ungetregister32(R_EDI);                                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                                   inc(tempreference.offset,4);                                   emit_ref_reg(A_MOV,S_L,                                     newreference(tempreference),R_EDI);                                   r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize+4);                                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,r)));{$ifndef noAllocEdi}                                   ungetregister32(R_EDI);{$endif noAllocEdi}                                 end                               else                                 begin                                   inc(tempreference.offset,4);                                   emit_push_mem(tempreference);                                   dec(tempreference.offset,4);                                   emit_push_mem(tempreference);                                 end;                             end;                         4 : begin                               inc(pushedparasize,4);                               if inlined then                                 begin{$ifndef noAllocEdi}                                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                                   emit_ref_reg(A_MOV,S_L,                                     newreference(tempreference),R_EDI);                                   r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,r)));{$ifndef noAllocEdi}                                   ungetregister32(R_EDI);{$endif noAllocEdi}                                 end                               else                                 emit_push_mem(tempreference);                             end;                       1,2 : begin                               if alignment=4 then                                begin                                  opsize:=S_L;                                  hreg:=R_EDI;                                  inc(pushedparasize,4);                                end                               else                                begin                                  opsize:=S_W;                                  hreg:=R_DI;                                  inc(pushedparasize,2);                                end;                               if inlined then                                begin{$ifndef noAllocEdi}                                  getexplicitregister32(R_EDI);{$endif noAllocEdi}                                  emit_ref_reg(A_MOV,opsize,                                    newreference(tempreference),hreg);                                  r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                  exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,opsize,hreg,r)));{$ifndef noAllocEdi}                                  ungetregister32(R_EDI);{$endif noAllocEdi}                                end                               else                                exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,opsize,                                  newreference(tempreference))));                             end;                           else                             internalerror(234231);                        end;                      end;                    floatdef :                      begin                        case pfloatdef(p^.resulttype)^.typ of                          f32bit,                          s32real :                            begin                               inc(pushedparasize,4);                               if inlined then                                 begin{$ifndef noAllocEdi}                                    getexplicitregister32(R_EDI);{$endif noAllocEdi}                                    emit_ref_reg(A_MOV,S_L,                                      newreference(tempreference),R_EDI);                                    r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                    exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,r)));{$ifndef noAllocEdi}                                    ungetregister32(R_EDI);{$endif noAllocEdi}                                 end                               else                                 emit_push_mem(tempreference);                            end;                          s64real,                          s64comp :                            begin                              inc(pushedparasize,4);                              inc(tempreference.offset,4);                              if inlined then                                begin{$ifndef noAllocEdi}                                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                                   emit_ref_reg(A_MOV,S_L,                                     newreference(tempreference),R_EDI);                                   r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,r)));{$ifndef noAllocEdi}                                   ungetregister32(R_EDI);{$endif noAllocEdi}                                end                              else                                emit_push_mem(tempreference);                              inc(pushedparasize,4);                              dec(tempreference.offset,4);                              if inlined then                                begin{$ifndef noAllocEdi}                                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                                   emit_ref_reg(A_MOV,S_L,                                     newreference(tempreference),R_EDI);                                   r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,r)));{$ifndef noAllocEdi}                                   ungetregister32(R_EDI);{$endif noAllocEdi}                                end                              else                                emit_push_mem(tempreference);                            end;                          s80real :                            begin                              inc(pushedparasize,4);                              if alignment=4 then                                inc(tempreference.offset,8)                              else                                inc(tempreference.offset,6);                              if inlined then                                begin{$ifndef noAllocEdi}                                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                                   emit_ref_reg(A_MOV,S_L,                                     newreference(tempreference),R_EDI);                                   r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,r)));{$ifndef noAllocEdi}                                   ungetregister32(R_EDI);{$endif noAllocEdi}                                end                              else                                emit_push_mem(tempreference);                              dec(tempreference.offset,4);                              inc(pushedparasize,4);                              if inlined then                                begin{$ifndef noAllocEdi}                                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                                   emit_ref_reg(A_MOV,S_L,                                     newreference(tempreference),R_EDI);                                   r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,r)));{$ifndef noAllocEdi}                                   ungetregister32(R_EDI);{$endif noAllocEdi}                                end                              else                                emit_push_mem(tempreference);                              if alignment=4 then                                begin                                  opsize:=S_L;                                  hreg:=R_EDI;                                  inc(pushedparasize,4);                                  dec(tempreference.offset,4);                                end                              else                                begin                                  opsize:=S_W;                                  hreg:=R_DI;                                  inc(pushedparasize,2);                                  dec(tempreference.offset,2);                                end;                              if inlined then                                begin{$ifndef noAllocEdi}                                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                                   emit_ref_reg(A_MOV,opsize,                                     newreference(tempreference),hreg);                                   r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,opsize,hreg,r)));{$ifndef noAllocEdi}                                   ungetregister32(R_EDI);{$endif noAllocEdi}                                end                              else                                exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,opsize,                                  newreference(tempreference))));                          end;                        end;                      end;                    pointerdef,                    procvardef,                    classrefdef:                      begin                         inc(pushedparasize,4);                         if inlined then                           begin{$ifndef noAllocEdi}                              getexplicitregister32(R_EDI);{$endif noAllocEdi}                              emit_ref_reg(A_MOV,S_L,                                newreference(tempreference),R_EDI);                              r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                              exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,r)));{$ifndef noAllocEdi}                              ungetregister32(R_EDI);{$endif noAllocEdi}                           end                         else                           emit_push_mem(tempreference);                      end;                    arraydef,                    recorddef,                    stringdef,                    setdef,                    objectdef :                      begin                         { even some structured types are 32 bit }                         if is_widestring(p^.resulttype) or                            is_ansistring(p^.resulttype) or                            is_smallset(p^.resulttype) or                            ((p^.resulttype^.deftype in [recorddef,arraydef]) and (p^.resulttype^.size<=4)                             and ((p^.resulttype^.deftype<>arraydef) or not                              (parraydef(p^.resulttype)^.IsConstructor or                               parraydef(p^.resulttype)^.isArrayOfConst or                               is_open_array(p^.resulttype)))                            ) or                            ((p^.resulttype^.deftype=objectdef) and                             pobjectdef(p^.resulttype)^.is_class) then                           begin                              if (p^.resulttype^.size>2) or                                 ((alignment=4) and (p^.resulttype^.size>0)) then                                begin                                  inc(pushedparasize,4);                                  if inlined then                                    begin                                      r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                      concatcopy(tempreference,r^,4,false,false);                                    end                                  else                                    emit_push_mem(tempreference);                                end                              else                                begin                                  if p^.resulttype^.size>0 then                                    begin                                      inc(pushedparasize,2);                                      if inlined then                                        begin                                          r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                                          concatcopy(tempreference,r^,2,false,false);                                        end                                      else                                        exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,S_W,newreference(tempreference))));                                    end;                                end;                           end                         { call by value open array ? }                         else if is_cdecl then                           begin                             { push on stack }                             size:=align(p^.resulttype^.size,alignment);                             inc(pushedparasize,size);                             emit_const_reg(A_SUB,S_L,size,R_ESP);                             r:=new_reference(R_ESP,0);                             concatcopy(tempreference,r^,size,false,false);                           end                         else                           internalerror(8954);                      end;                    else                      CGMessage(cg_e_illegal_expression);                  end;               end;             LOC_JUMP:               begin                  getlabel(hlabel);                  if alignment=4 then                   begin                     opsize:=S_L;                     inc(pushedparasize,4);                   end                  else                   begin                     opsize:=S_W;                     inc(pushedparasize,2);                   end;                  emitlab(truelabel);                  if inlined then                    begin                       r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                       emit_const_ref(A_MOV,opsize,1,r);                    end                  else                    exprasmlist^.concat(new(paicpu,op_const(A_PUSH,opsize,1)));                  emitjmp(C_None,hlabel);                  emitlab(falselabel);                  if inlined then                    begin                       r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                       emit_const_ref(A_MOV,opsize,0,r);                    end                  else                    exprasmlist^.concat(new(paicpu,op_const(A_PUSH,opsize,0)));                  emitlab(hlabel);               end;             LOC_FLAGS:               begin                  if not(R_EAX in unused) then                    begin{$ifndef noAllocEdi}                      getexplicitregister32(R_EDI);{$endif noAllocEdi}                      emit_reg_reg(A_MOV,S_L,R_EAX,R_EDI);                    end;                  emit_flag2reg(p^.location.resflags,R_AL);                  emit_reg_reg(A_MOVZX,S_BW,R_AL,R_AX);                  if alignment=4 then                   begin                     opsize:=S_L;                     hreg:=R_EAX;                     inc(pushedparasize,4);                   end                  else                   begin                     opsize:=S_W;                     hreg:=R_AX;                     inc(pushedparasize,2);                   end;                  if inlined then                    begin                       r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                       exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,opsize,hreg,r)));                    end                  else                    exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,opsize,hreg)));                  if not(R_EAX in unused) then                    begin                      emit_reg_reg(A_MOV,S_L,R_EDI,R_EAX);{$ifndef noAllocEdi}                      ungetregister32(R_EDI);{$endif noAllocEdi}                    end;               end;{$ifdef SUPPORT_MMX}             LOC_MMXREGISTER,             LOC_CMMXREGISTER:               begin                  inc(pushedparasize,8); { was missing !!! (PM) }                  emit_const_reg(                    A_SUB,S_L,8,R_ESP);{$ifdef GDB}                  if (cs_debuginfo in aktmoduleswitches) and                     (exprasmlist^.first=exprasmlist^.last) then                    exprasmlist^.concat(new(pai_force_line,init));{$endif GDB}                  if inlined then                    begin                       r:=new_reference(procinfo^.framepointer,para_offset-pushedparasize);                       exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOVQ,S_NO,                         p^.location.register,r)));                    end                  else                     begin                        r:=new_reference(R_ESP,0);                        exprasmlist^.concat(new(paicpu,op_reg_ref(                          A_MOVQ,S_NO,p^.location.register,r)));                     end;               end;{$endif SUPPORT_MMX}          end;      end;{*****************************************************************************                           Emit Float Functions*****************************************************************************}    procedure floatloadops(t : tfloattype;var op : tasmop;var s : topsize);      begin         case t of            s32real : begin                         op:=A_FLD;                         s:=S_FS;                      end;            s64real : begin                         op:=A_FLD;                         { ???? }                         s:=S_FL;                      end;            s80real : begin                         op:=A_FLD;                         s:=S_FX;                      end;            s64comp : begin                         op:=A_FILD;                         s:=S_IQ;                      end;            else internalerror(17);         end;      end;    procedure floatload(t : tfloattype;const ref : treference);      var         op : tasmop;         s : topsize;      begin         floatloadops(t,op,s);         exprasmlist^.concat(new(paicpu,op_ref(op,s,           newreference(ref))));         inc(fpuvaroffset);      end;    procedure floatstoreops(t : tfloattype;var op : tasmop;var s : topsize);      begin         case t of            s32real : begin                         op:=A_FSTP;                         s:=S_FS;                      end;            s64real : begin                         op:=A_FSTP;                         s:=S_FL;                      end;            s80real : begin                         op:=A_FSTP;                          s:=S_FX;                      end;            s64comp : begin                         op:=A_FISTP;                         s:=S_IQ;                      end;         else           internalerror(17);         end;      end;    procedure floatstore(t : tfloattype;const ref : treference);      var         op : tasmop;         s : topsize;      begin         floatstoreops(t,op,s);         exprasmlist^.concat(new(paicpu,op_ref(op,s,           newreference(ref))));         dec(fpuvaroffset);      end;{*****************************************************************************                           Emit Functions*****************************************************************************}    procedure maketojumpbool(p : ptree);    {      produces jumps to true respectively false labels using boolean expressions    }      var        opsize : topsize;        storepos : tfileposinfo;      begin         if p^.error then           exit;         storepos:=aktfilepos;         aktfilepos:=p^.fileinfo;         if is_boolean(p^.resulttype) then           begin              if is_constboolnode(p) then                begin                   if p^.value<>0 then                     emitjmp(C_None,truelabel)                   else                     emitjmp(C_None,falselabel);                end              else                begin                   opsize:=def_opsize(p^.resulttype);                   case p^.location.loc of                      LOC_CREGISTER,LOC_REGISTER : begin                                        emit_reg_reg(A_OR,opsize,p^.location.register,                                          p^.location.register);                                        ungetregister(p^.location.register);                                        emitjmp(C_NZ,truelabel);                                        emitjmp(C_None,falselabel);                                     end;                      LOC_MEM,LOC_REFERENCE : begin                                        emit_const_ref(                                          A_CMP,opsize,0,newreference(p^.location.reference));                                        del_reference(p^.location.reference);                                        emitjmp(C_NZ,truelabel);                                        emitjmp(C_None,falselabel);                                     end;                      LOC_FLAGS : begin                                     emitjmp(flag_2_cond[p^.location.resflags],truelabel);                                     emitjmp(C_None,falselabel);                                  end;                   end;                end;           end         else           CGMessage(type_e_mismatch);         aktfilepos:=storepos;      end;    { produces if necessary overflowcode }    procedure emitoverflowcheck(p:ptree);      var         hl : pasmlabel;      begin         if not(cs_check_overflow in aktlocalswitches) then          exit;         getlabel(hl);         if not ((p^.resulttype^.deftype=pointerdef) or                ((p^.resulttype^.deftype=orddef) and                 (porddef(p^.resulttype)^.typ in [u64bit,u16bit,u32bit,u8bit,uchar,                                                  bool8bit,bool16bit,bool32bit]))) then           emitjmp(C_NO,hl)         else           emitjmp(C_NB,hl);         emitcall('FPC_OVERFLOW');         emitlab(hl);      end;    { produces range check code, while one of the operands is a 64 bit      integer }    procedure emitrangecheck64(p : ptree;todef : pdef);      begin         CGMessage(cg_w_64bit_range_check_not_supported);         {internalerror(28699);}      end;     { produces if necessary rangecheckcode }     procedure emitrangecheck(p:ptree;todef:pdef);     {       generate range checking code for the value at location t. The       type used is the checked against todefs ranges. fromdef (p.resulttype)       is the original type used at that location, when both defs are       equal the check is also insert (needed for succ,pref,inc,dec)     }      var        neglabel,        poslabel : pasmlabel;        href   : treference;        rstr   : string;        hreg   : tregister;        opsize : topsize;        op     : tasmop;        fromdef : pdef;        lto,hto,        lfrom,hfrom : longint;        doublebound,        is_reg,        popecx : boolean;      begin        { range checking on and range checkable value? }        if not(cs_check_range in aktlocalswitches) or           not(todef^.deftype in [orddef,enumdef,arraydef]) then          exit;        { only check when assigning to scalar, subranges are different,          when todef=fromdef then the check is always generated }        fromdef:=p^.resulttype;        if is_64bitint(fromdef) or is_64bitint(todef) then          begin             emitrangecheck64(p,todef);             exit;          end;        {we also need lto and hto when checking if we need to use doublebound!        (JM)}        getrange(todef,lto,hto);        if todef<>fromdef then         begin           getrange(p^.resulttype,lfrom,hfrom);           { first check for not being u32bit, then if the to is bigger than             from }           if (lto<hto) and (lfrom<hfrom) and              (lto<=lfrom) and (hto>=hfrom) then            exit;         end;        { generate the rangecheck code for the def where we are going to          store the result }        doublebound:=false;        case todef^.deftype of          orddef :            begin              porddef(todef)^.genrangecheck;              rstr:=porddef(todef)^.getrangecheckstring;              doublebound:=(porddef(todef)^.typ=u32bit) and (lto>hto);            end;          enumdef :            begin              penumdef(todef)^.genrangecheck;              rstr:=penumdef(todef)^.getrangecheckstring;            end;          arraydef :            begin              parraydef(todef)^.genrangecheck;              rstr:=parraydef(todef)^.getrangecheckstring;              doublebound:=(lto>hto);            end;        end;      { get op and opsize }        opsize:=def2def_opsize(fromdef,u32bitdef);        if opsize in [S_B,S_W,S_L] then         op:=A_MOV        else         if is_signed(fromdef) then          op:=A_MOVSX         else          op:=A_MOVZX;        is_reg:=(p^.location.loc in [LOC_REGISTER,LOC_CREGISTER]);        if is_reg then          hreg:=p^.location.register;        if not target_os.use_bound_instruction then         begin           { FPC_BOUNDCHECK needs to be called with              %ecx - value              %edi - pointer to the ranges }           popecx:=false;           if not(is_reg) or              (p^.location.register<>R_ECX) then            begin              if not(R_ECX in unused) then               begin                 exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,R_ECX)));                 popecx:=true;               end                 else exprasmlist^.concat(new(pairegalloc,alloc(R_ECX)));              if is_reg then               emit_reg_reg(op,opsize,p^.location.register,R_ECX)              else               emit_ref_reg(op,opsize,newreference(p^.location.reference),R_ECX);            end;           if doublebound then            begin              getlabel(neglabel);              getlabel(poslabel);              emit_reg_reg(A_OR,S_L,R_ECX,R_ECX);              emitjmp(C_L,neglabel);            end;           { insert bound instruction only }{$ifndef noAllocEdi}           getexplicitregister32(R_EDI);{$endif noAllocEdi}           exprasmlist^.concat(new(paicpu,op_sym_ofs_reg(A_MOV,S_L,newasmsymbol(rstr),0,R_EDI)));           emitcall('FPC_BOUNDCHECK');{$ifndef noAllocEdi}           ungetregister32(R_EDI);{$endif noAllocEdi}           { u32bit needs 2 checks }           if doublebound then            begin              emitjmp(C_None,poslabel);              emitlab(neglabel);{$ifndef noAllocEdi}              getexplicitregister32(R_EDI);{$endif noAllocEdi}              exprasmlist^.concat(new(paicpu,op_sym_ofs_reg(A_MOV,S_L,newasmsymbol(rstr),8,R_EDI)));              emitcall('FPC_BOUNDCHECK');{$ifndef noAllocEdi}              ungetregister32(R_EDI);{$endif noAllocEdi}              emitlab(poslabel);            end;           if popecx then            exprasmlist^.concat(new(paicpu,op_reg(A_POP,S_L,R_ECX)))           else exprasmlist^.concat(new(pairegalloc,dealloc(R_ECX)));         end        else         begin           reset_reference(href);           href.symbol:=newasmsymbol(rstr);           { load the value in a register }           if is_reg then            begin              { be sure that hreg is a 32 bit reg, if not load it in %edi }              if p^.location.register in [R_EAX..R_EDI] then               hreg:=p^.location.register              else               begin{$ifndef noAllocEdi}                 getexplicitregister32(R_EDI);{$endif noAllocEdi}                 emit_reg_reg(op,opsize,p^.location.register,R_EDI);                 hreg:=R_EDI;               end;            end           else            begin{$ifndef noAllocEdi}              getexplicitregister32(R_EDI);{$endif noAllocEdi}              emit_ref_reg(op,opsize,newreference(p^.location.reference),R_EDI);              hreg:=R_EDI;            end;           if doublebound then            begin              getlabel(neglabel);              getlabel(poslabel);              emit_reg_reg(A_TEST,S_L,hreg,hreg);              emitjmp(C_L,neglabel);            end;           { insert bound instruction only }           exprasmlist^.concat(new(paicpu,op_reg_ref(A_BOUND,S_L,hreg,newreference(href))));           { u32bit needs 2 checks }           if doublebound then            begin              href.offset:=8;              emitjmp(C_None,poslabel);              emitlab(neglabel);              exprasmlist^.concat(new(paicpu,op_reg_ref(A_BOUND,S_L,hreg,newreference(href))));              emitlab(poslabel);            end;{$ifndef noAllocEdi}           if hreg = R_EDI then             ungetregister32(R_EDI);{$endif noAllocEdi}         end;      end;    procedure concatcopy(source,dest : treference;size : longint;delsource,loadref : boolean);      const         isizes : array[0..3] of topsize=(S_L,S_B,S_W,S_B);         ishr : array[0..3] of byte=(2,0,1,0);      var         ecxpushed : boolean;         helpsize : longint;         i : byte;         reg8,reg32 : tregister;         swap : boolean;         procedure maybepushecx;         begin           if not(R_ECX in unused) then             begin               exprasmlist^.concat(new(paicpu,op_reg(A_PUSH,S_L,R_ECX)));               ecxpushed:=true;             end;         end;      begin{$IfNDef regallocfix}        If delsource then           del_reference(source);{$EndIf regallocfix}         if (not loadref) and            ((size<=8) or             (not(cs_littlesize in aktglobalswitches ) and (size<=12))) then           begin              helpsize:=size shr 2;{$ifndef noAllocEdi}              getexplicitregister32(R_EDI);{$endif noAllocEdi}              for i:=1 to helpsize do                begin                   emit_ref_reg(A_MOV,S_L,newreference(source),R_EDI);{$ifdef regallocfix}                   If (size = 4) and delsource then                     del_reference(source);{$endif regallocfix}                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDI,newreference(dest))));                   inc(source.offset,4);                   inc(dest.offset,4);                   dec(size,4);                end;              if size>1 then                begin                   emit_ref_reg(A_MOV,S_W,newreference(source),R_DI);{$ifdef regallocfix}                   If (size = 2) and delsource then                     del_reference(source);{$endif regallocfix}                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_W,R_DI,newreference(dest))));                   inc(source.offset,2);                   inc(dest.offset,2);                   dec(size,2);                end;{$ifndef noAllocEdi}              ungetregister32(R_EDI);{$endif noAllocEdi}              if size>0 then                begin                   { and now look for an 8 bit register }                   swap:=false;                   if R_EAX in unused then reg8:=R_AL                   else if R_EBX in unused then reg8:=R_BL                   else if R_ECX in unused then reg8:=R_CL                   else if R_EDX in unused then reg8:=R_DL                   else                      begin                         swap:=true;                         { we need only to check 3 registers, because }                         { one is always not index or base          }                         if (dest.base<>R_EAX) and (dest.index<>R_EAX) then                           begin                              reg8:=R_AL;                              reg32:=R_EAX;                           end                         else if (dest.base<>R_EBX) and (dest.index<>R_EBX) then                           begin                              reg8:=R_BL;                              reg32:=R_EBX;                           end                         else if (dest.base<>R_ECX) and (dest.index<>R_ECX) then                           begin                              reg8:=R_CL;                              reg32:=R_ECX;                           end;                      end;                   if swap then                     { was earlier XCHG, of course nonsense }                     begin{$ifndef noAllocEdi}                       getexplicitregister32(R_EDI);{$endif noAllocEdi}                       emit_reg_reg(A_MOV,S_L,reg32,R_EDI);                     end;                   emit_ref_reg(A_MOV,S_B,newreference(source),reg8);{$ifdef regallocfix}                   If delsource then                     del_reference(source);{$endif regallocfix}                   exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_B,reg8,newreference(dest))));                   if swap then                     begin                       emit_reg_reg(A_MOV,S_L,R_EDI,reg32);{$ifndef noAllocEdi}                       ungetregister32(R_EDI);{$endif noAllocEdi}                     end;                end;           end         else           begin{$ifndef noAllocEdi}              getexplicitregister32(R_EDI);{$endif noAllocEdi}              emit_ref_reg(A_LEA,S_L,newreference(dest),R_EDI);{$ifdef regallocfix}             {is this ok?? (JM)}              del_reference(dest);{$endif regallocfix}{$ifndef noAllocEdi}              exprasmlist^.concat(new(pairegalloc,alloc(R_ESI)));{$endif noAllocEdi}              if loadref then                emit_ref_reg(A_MOV,S_L,newreference(source),R_ESI)              else                begin                  emit_ref_reg(A_LEA,S_L,newreference(source),R_ESI);{$ifdef regallocfix}                  if delsource then                    del_reference(source);{$endif regallocfix}                end;              exprasmlist^.concat(new(paicpu,op_none(A_CLD,S_NO)));              ecxpushed:=false;              if cs_littlesize in aktglobalswitches  then                begin                   maybepushecx;                   emit_const_reg(A_MOV,S_L,size,R_ECX);                   exprasmlist^.concat(new(paicpu,op_none(A_REP,S_NO)));                   exprasmlist^.concat(new(paicpu,op_none(A_MOVSB,S_NO)));                end              else                begin                   helpsize:=size shr 2;                   size:=size and 3;                   if helpsize>1 then                    begin                      maybepushecx;                      emit_const_reg(A_MOV,S_L,helpsize,R_ECX);                      exprasmlist^.concat(new(paicpu,op_none(A_REP,S_NO)));                    end;                   if helpsize>0 then                    exprasmlist^.concat(new(paicpu,op_none(A_MOVSD,S_NO)));                   if size>1 then                     begin                        dec(size,2);                        exprasmlist^.concat(new(paicpu,op_none(A_MOVSW,S_NO)));                     end;                   if size=1 then                     exprasmlist^.concat(new(paicpu,op_none(A_MOVSB,S_NO)));                end;{$ifndef noAllocEdi}              ungetregister32(R_EDI);              exprasmlist^.concat(new(pairegalloc,dealloc(R_ESI)));{$endif noAllocEdi}              if ecxpushed then                begin                  exprasmlist^.concat(new(paicpu,op_reg(A_POP,S_L,R_ECX)));                end;              { loading SELF-reference again }              maybe_loadesi;           end;         if delsource then           ungetiftemp(source);      end;    procedure emitloadord2reg(const location:Tlocation;orddef:Porddef;                              destreg:Tregister;delloc:boolean);    {A lot smaller and less bug sensitive than the original unfolded loads.}    var tai:Paicpu;        r:Preference;    begin        tai := nil;        case location.loc of            LOC_REGISTER,LOC_CREGISTER:                begin                    case orddef^.typ of                        u8bit:                            tai:=new(paicpu,op_reg_reg(A_MOVZX,S_BL,location.register,destreg));                        s8bit:                            tai:=new(paicpu,op_reg_reg(A_MOVSX,S_BL,location.register,destreg));                        u16bit:                            tai:=new(paicpu,op_reg_reg(A_MOVZX,S_WL,location.register,destreg));                        s16bit:                            tai:=new(paicpu,op_reg_reg(A_MOVSX,S_WL,location.register,destreg));                        u32bit,s32bit:                            if location.register <> destreg then                              tai:=new(paicpu,op_reg_reg(A_MOV,S_L,location.register,destreg));                    end;                    if delloc then                        ungetregister(location.register);                end;            LOC_MEM,            LOC_REFERENCE:                begin                    if location.reference.is_immediate then                     tai:=new(paicpu,op_const_reg(A_MOV,S_L,location.reference.offset,destreg))                    else                     begin                       r:=newreference(location.reference);                       case orddef^.typ of                         u8bit:                            tai:=new(paicpu,op_ref_reg(A_MOVZX,S_BL,r,destreg));                         s8bit:                            tai:=new(paicpu,op_ref_reg(A_MOVSX,S_BL,r,destreg));                         u16bit:                            tai:=new(paicpu,op_ref_reg(A_MOVZX,S_WL,r,destreg));                         s16bit:                            tai:=new(paicpu,op_ref_reg(A_MOVSX,S_WL,r,destreg));                         u32bit:                            tai:=new(paicpu,op_ref_reg(A_MOV,S_L,r,destreg));                         s32bit:                            tai:=new(paicpu,op_ref_reg(A_MOV,S_L,r,destreg));                       end;                     end;                    if delloc then                        del_reference(location.reference);                end            else                internalerror(6);        end;        if assigned(tai) then          exprasmlist^.concat(tai);    end;    { if necessary ESI is reloaded after a call}    procedure maybe_loadesi;      var         hp : preference;         p : pprocinfo;         i : longint;      begin         if assigned(procinfo^._class) then           begin{$ifndef noAllocEdi}              exprasmlist^.concat(new(pairegalloc,alloc(R_ESI)));{$endif noAllocEdi}              if lexlevel>normal_function_level then                begin                   new(hp);                   reset_reference(hp^);                   hp^.offset:=procinfo^.framepointer_offset;                   hp^.base:=procinfo^.framepointer;                   emit_ref_reg(A_MOV,S_L,hp,R_ESI);                   p:=procinfo^.parent;                   for i:=3 to lexlevel-1 do                     begin                        new(hp);                        reset_reference(hp^);                        hp^.offset:=p^.framepointer_offset;                        hp^.base:=R_ESI;                        emit_ref_reg(A_MOV,S_L,hp,R_ESI);                        p:=p^.parent;                     end;                   new(hp);                   reset_reference(hp^);                   hp^.offset:=p^.selfpointer_offset;                   hp^.base:=R_ESI;                   emit_ref_reg(A_MOV,S_L,hp,R_ESI);                end              else                begin                   new(hp);                   reset_reference(hp^);                   hp^.offset:=procinfo^.selfpointer_offset;                   hp^.base:=procinfo^.framepointer;                   emit_ref_reg(A_MOV,S_L,hp,R_ESI);                end;           end;      end;   { DO NOT RELY on the fact that the ptree is not yet swaped     because of inlining code PM }    procedure firstcomplex(p : ptree);      var         hp : ptree;      begin         { always calculate boolean AND and OR from left to right }         if (p^.treetype in [orn,andn]) and            (p^.left^.resulttype^.deftype=orddef) and            (porddef(p^.left^.resulttype)^.typ in [bool8bit,bool16bit,bool32bit]) then           begin             { p^.swaped:=false}             if p^.swaped then               internalerror(234234);           end         else           if (p^.left^.registers32<p^.right^.registers32) and           { the following check is appropriate, because all }           { 4 registers are rarely used and it is thereby   }           { achieved that the extra code is being dropped   }           { by exchanging not commutative operators     }              (p^.right^.registers32<=4) then            begin              hp:=p^.left;              p^.left:=p^.right;              p^.right:=hp;              p^.swaped:=not p^.swaped;            end;         {else           p^.swaped:=false; do not modify }      end;{*****************************************************************************                            Entry/Exit Code Functions*****************************************************************************}  procedure genprofilecode;    var      pl : pasmlabel;    begin      if (po_assembler in aktprocsym^.definition^.procoptions) then       exit;      case target_info.target of         target_i386_linux:           begin              getlabel(pl);              emitcall('mcount');              exprasmlist^.insert(new(paicpu,op_sym_ofs_reg(A_MOV,S_L,pl,0,R_EDX)));              exprasmlist^.insert(new(pai_section,init(sec_code)));              exprasmlist^.insert(new(pai_const,init_32bit(0)));              exprasmlist^.insert(new(pai_label,init(pl)));              exprasmlist^.insert(new(pai_align,init(4)));              exprasmlist^.insert(new(pai_section,init(sec_data)));           end;         target_i386_go32v2:           begin              emitinsertcall('MCOUNT');           end;      end;    end;    procedure generate_interrupt_stackframe_entry;      begin         { save the registers of an interrupt procedure }         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_EAX)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_EBX)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_ECX)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_EDX)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_ESI)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_EDI)));         { .... also the segment registers }         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_W,R_DS)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_W,R_ES)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_W,R_FS)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_W,R_GS)));      end;    procedure generate_interrupt_stackframe_exit;      begin         { restore the registers of an interrupt procedure }         { this was all with entrycode instead of exitcode !!}         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_L,R_EAX)));         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_L,R_EBX)));         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_L,R_ECX)));         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_L,R_EDX)));         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_L,R_ESI)));         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_L,R_EDI)));         { .... also the segment registers }         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_W,R_DS)));         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_W,R_ES)));         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_W,R_FS)));         procinfo^.aktexitcode^.concat(new(paicpu,op_reg(A_POP,S_W,R_GS)));        { this restores the flags }         procinfo^.aktexitcode^.concat(new(paicpu,op_none(A_IRET,S_NO)));      end;  { generates the code for threadvar initialisation }  procedure initialize_threadvar(p : pnamedindexobject);{$ifndef FPC}far;{$endif}    var       hr : treference;    begin       if (psym(p)^.typ=varsym) and          (vo_is_thread_var in pvarsym(p)^.varoptions) then         begin            exprasmlist^.concat(new(paicpu,op_const(A_PUSH,S_L,pvarsym(p)^.getsize)));            reset_reference(hr);            hr.symbol:=newasmsymbol(pvarsym(p)^.mangledname);            emitpushreferenceaddr(hr);            emitcall('FPC_INIT_THREADVAR');         end;    end;    { initilizes data of type t                           }    { if is_already_ref is true then the routines assumes }    { that r points to the data to initialize             }    procedure initialize(t : pdef;const ref : treference;is_already_ref : boolean);      var         hr : treference;      begin         if is_ansistring(t) or           is_widestring(t) then           begin              emit_const_ref(A_MOV,S_L,0,                newreference(ref));           end         else           begin              reset_reference(hr);              hr.symbol:=t^.get_inittable_label;              emitpushreferenceaddr(hr);              if is_already_ref then                exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,S_L,                  newreference(ref))))              else                emitpushreferenceaddr(ref);              emitcall('FPC_INITIALIZE');           end;      end;    { finalizes data of type t                            }    { if is_already_ref is true then the routines assumes }    { that r points to the data to finalizes              }    procedure finalize(t : pdef;const ref : treference;is_already_ref : boolean);      var         r : treference;      begin         if is_ansistring(t) or           is_widestring(t) then           begin              decrstringref(t,ref);           end         else           begin              reset_reference(r);              r.symbol:=t^.get_inittable_label;              emitpushreferenceaddr(r);              if is_already_ref then                exprasmlist^.concat(new(paicpu,op_ref(A_PUSH,S_L,                  newreference(ref))))              else                emitpushreferenceaddr(ref);              emitcall('FPC_FINALIZE');           end;      end;  { generates the code for initialisation of local data }  procedure initialize_data(p : pnamedindexobject);{$ifndef FPC}far;{$endif}    var       hr : treference;    begin       if (psym(p)^.typ=varsym) and          assigned(pvarsym(p)^.vartype.def) and          not((pvarsym(p)^.vartype.def^.deftype=objectdef) and            pobjectdef(pvarsym(p)^.vartype.def)^.is_class) and          pvarsym(p)^.vartype.def^.needs_inittable then         begin            if assigned(procinfo) then              procinfo^.flags:=procinfo^.flags or pi_needs_implicit_finally;            reset_reference(hr);            if psym(p)^.owner^.symtabletype in [localsymtable,inlinelocalsymtable] then              begin                 hr.base:=procinfo^.framepointer;                 hr.offset:=-pvarsym(p)^.address+pvarsym(p)^.owner^.address_fixup;              end            else              begin                 hr.symbol:=newasmsymbol(pvarsym(p)^.mangledname);              end;            initialize(pvarsym(p)^.vartype.def,hr,false);         end;    end;  { generates the code for incrementing the reference count of parameters }  procedure incr_data(p : pnamedindexobject);{$ifndef FPC}far;{$endif}    var       hr : treference;    begin       if (psym(p)^.typ=varsym) and          not((pvarsym(p)^.vartype.def^.deftype=objectdef) and            pobjectdef(pvarsym(p)^.vartype.def)^.is_class) and          pvarsym(p)^.vartype.def^.needs_inittable and          ((pvarsym(p)^.varspez=vs_value) {or           (pvarsym(p)^.varspez=vs_const) and           not(dont_copy_const_param(pvarsym(p)^.definition))}) then         begin            procinfo^.flags:=procinfo^.flags or pi_needs_implicit_finally;            reset_reference(hr);            hr.symbol:=pvarsym(p)^.vartype.def^.get_inittable_label;            emitpushreferenceaddr(hr);            reset_reference(hr);            hr.base:=procinfo^.framepointer;            hr.offset:=pvarsym(p)^.address+procinfo^.para_offset;            emitpushreferenceaddr(hr);            reset_reference(hr);            emitcall('FPC_ADDREF');         end;    end;  { generates the code for finalisation of local data }  procedure finalize_data(p : pnamedindexobject);{$ifndef FPC}far;{$endif}    var       hr : treference;    begin       if (psym(p)^.typ=varsym) and          assigned(pvarsym(p)^.vartype.def) and          not((pvarsym(p)^.vartype.def^.deftype=objectdef) and          pobjectdef(pvarsym(p)^.vartype.def)^.is_class) and          pvarsym(p)^.vartype.def^.needs_inittable then         begin            { not all kind of parameters need to be finalized  }            if (psym(p)^.owner^.symtabletype=parasymtable) and              ((pvarsym(p)^.varspez=vs_var)  or               (pvarsym(p)^.varspez=vs_const) { and               (dont_copy_const_param(pvarsym(p)^.definition)) } ) then              exit;            if assigned(procinfo) then              procinfo^.flags:=procinfo^.flags or pi_needs_implicit_finally;            reset_reference(hr);            case psym(p)^.owner^.symtabletype of               localsymtable,inlinelocalsymtable:                 begin                    hr.base:=procinfo^.framepointer;                    hr.offset:=-pvarsym(p)^.address+pvarsym(p)^.owner^.address_fixup;                 end;               parasymtable,inlineparasymtable:                 begin                    hr.base:=procinfo^.framepointer;                    hr.offset:=pvarsym(p)^.address+procinfo^.para_offset;                 end;               else                 hr.symbol:=newasmsymbol(pvarsym(p)^.mangledname);            end;            finalize(pvarsym(p)^.vartype.def,hr,false);         end;    end;  { generates the code to make local copies of the value parameters }  procedure copyvalueparas(p : pnamedindexobject);{$ifndef fpc}far;{$endif}    var      href1,href2 : treference;      r    : preference;      power,len  : longint;      opsize : topsize;      again,ok : pasmlabel;    begin       if (psym(p)^.typ=varsym) and          (pvarsym(p)^.varspez=vs_value) and          (push_addr_param(pvarsym(p)^.vartype.def)) then        begin          if is_open_array(pvarsym(p)^.vartype.def) or             is_array_of_const(pvarsym(p)^.vartype.def) then           begin              { get stack space }              new(r);              reset_reference(r^);              r^.base:=procinfo^.framepointer;              r^.offset:=pvarsym(p)^.address+4+procinfo^.para_offset;{$ifndef noAllocEdi}              getexplicitregister32(R_EDI);{$endif noAllocEdi}              exprasmlist^.concat(new(paicpu,                op_ref_reg(A_MOV,S_L,r,R_EDI)));              exprasmlist^.concat(new(paicpu,                op_reg(A_INC,S_L,R_EDI)));              if (parraydef(pvarsym(p)^.vartype.def)^.elesize<>1) then               begin                 if ispowerof2(parraydef(pvarsym(p)^.vartype.def)^.elesize, power) then                   exprasmlist^.concat(new(paicpu,                     op_const_reg(A_SHL,S_L,                       power,R_EDI)))                 else                   exprasmlist^.concat(new(paicpu,                     op_const_reg(A_IMUL,S_L,                     parraydef(pvarsym(p)^.vartype.def)^.elesize,R_EDI)));               end;{$ifndef NOTARGETWIN32}              { windows guards only a few pages for stack growing, }              { so we have to access every page first              }              if target_os.id=os_i386_win32 then                begin                   getlabel(again);                   getlabel(ok);                   emitlab(again);                   exprasmlist^.concat(new(paicpu,                     op_const_reg(A_CMP,S_L,winstackpagesize,R_EDI)));                   emitjmp(C_C,ok);                   exprasmlist^.concat(new(paicpu,                     op_const_reg(A_SUB,S_L,winstackpagesize-4,R_ESP)));                   exprasmlist^.concat(new(paicpu,                     op_reg(A_PUSH,S_L,R_EAX)));                   exprasmlist^.concat(new(paicpu,                     op_const_reg(A_SUB,S_L,winstackpagesize,R_EDI)));                   emitjmp(C_None,again);                   emitlab(ok);                   exprasmlist^.concat(new(paicpu,                     op_reg_reg(A_SUB,S_L,R_EDI,R_ESP)));{$ifndef noAllocEdi}                   ungetregister32(R_EDI);{$endif noAllocEdi}                   { now reload EDI }                   new(r);                   reset_reference(r^);                   r^.base:=procinfo^.framepointer;                   r^.offset:=pvarsym(p)^.address+4+procinfo^.para_offset;{$ifndef noAllocEdi}                   getexplicitregister32(R_EDI);{$endif noAllocEdi}                   exprasmlist^.concat(new(paicpu,                     op_ref_reg(A_MOV,S_L,r,R_EDI)));                   exprasmlist^.concat(new(paicpu,                     op_reg(A_INC,S_L,R_EDI)));                   if (parraydef(pvarsym(p)^.vartype.def)^.elesize<>1) then                    begin                      if ispowerof2(parraydef(pvarsym(p)^.vartype.def)^.elesize, power) then                        exprasmlist^.concat(new(paicpu,                          op_const_reg(A_SHL,S_L,                            power,R_EDI)))                      else                        exprasmlist^.concat(new(paicpu,                          op_const_reg(A_IMUL,S_L,                          parraydef(pvarsym(p)^.vartype.def)^.elesize,R_EDI)));                    end;                end              else{$endif NOTARGETWIN32}                exprasmlist^.concat(new(paicpu,                  op_reg_reg(A_SUB,S_L,R_EDI,R_ESP)));              { load destination }              exprasmlist^.concat(new(paicpu,                op_reg_reg(A_MOV,S_L,R_ESP,R_EDI)));              { don't destroy the registers! }              exprasmlist^.concat(new(paicpu,                op_reg(A_PUSH,S_L,R_ECX)));              exprasmlist^.concat(new(paicpu,                op_reg(A_PUSH,S_L,R_ESI)));              { load count }              new(r);              reset_reference(r^);              r^.base:=procinfo^.framepointer;              r^.offset:=pvarsym(p)^.address+4+procinfo^.para_offset;              exprasmlist^.concat(new(paicpu,                op_ref_reg(A_MOV,S_L,r,R_ECX)));              { load source }              new(r);              reset_reference(r^);              r^.base:=procinfo^.framepointer;              r^.offset:=pvarsym(p)^.address+procinfo^.para_offset;              exprasmlist^.concat(new(paicpu,                op_ref_reg(A_MOV,S_L,r,R_ESI)));              { scheduled .... }              exprasmlist^.concat(new(paicpu,                op_reg(A_INC,S_L,R_ECX)));              { calculate size }              len:=parraydef(pvarsym(p)^.vartype.def)^.elesize;              opsize:=S_B;              if (len and 3)=0 then               begin                 opsize:=S_L;                 len:=len shr 2;               end              else               if (len and 1)=0 then                begin                  opsize:=S_W;                  len:=len shr 1;                end;              if ispowerof2(len, power) then                exprasmlist^.concat(new(paicpu,                  op_const_reg(A_SHL,S_L,                    power,R_ECX)))              else                exprasmlist^.concat(new(paicpu,                op_const_reg(A_IMUL,S_L,len,R_ECX)));              exprasmlist^.concat(new(paicpu,                op_none(A_REP,S_NO)));              case opsize of                S_B : exprasmlist^.concat(new(paicpu,op_none(A_MOVSB,S_NO)));                S_W : exprasmlist^.concat(new(paicpu,op_none(A_MOVSW,S_NO)));                S_L : exprasmlist^.concat(new(paicpu,op_none(A_MOVSD,S_NO)));              end;{$ifndef noAllocEdi}              ungetregister32(R_EDI);{$endif noAllocEdi}              exprasmlist^.concat(new(paicpu,                op_reg(A_POP,S_L,R_ESI)));              exprasmlist^.concat(new(paicpu,                op_reg(A_POP,S_L,R_ECX)));              { patch the new address }              new(r);              reset_reference(r^);              r^.base:=procinfo^.framepointer;              r^.offset:=pvarsym(p)^.address+procinfo^.para_offset;              exprasmlist^.concat(new(paicpu,                op_reg_ref(A_MOV,S_L,R_ESP,r)));           end          else           if is_shortstring(pvarsym(p)^.vartype.def) then            begin              reset_reference(href1);              href1.base:=procinfo^.framepointer;              href1.offset:=pvarsym(p)^.address+procinfo^.para_offset;              reset_reference(href2);              href2.base:=procinfo^.framepointer;              href2.offset:=-pvarsym(p)^.localvarsym^.address+pvarsym(p)^.localvarsym^.owner^.address_fixup;              copyshortstring(href2,href1,pstringdef(pvarsym(p)^.vartype.def)^.len,true,false);            end           else            begin              reset_reference(href1);              href1.base:=procinfo^.framepointer;              href1.offset:=pvarsym(p)^.address+procinfo^.para_offset;              reset_reference(href2);              href2.base:=procinfo^.framepointer;              href2.offset:=-pvarsym(p)^.localvarsym^.address+pvarsym(p)^.localvarsym^.owner^.address_fixup;              concatcopy(href1,href2,pvarsym(p)^.vartype.def^.size,true,true);            end;        end;    end;  procedure inittempansistrings;    var       hp : ptemprecord;       r : preference;    begin       hp:=templist;       while assigned(hp) do         begin           if hp^.temptype in [tt_ansistring,tt_freeansistring] then            begin              procinfo^.flags:=procinfo^.flags or pi_needs_implicit_finally;              new(r);              reset_reference(r^);              r^.base:=procinfo^.framepointer;              r^.offset:=hp^.pos;              emit_const_ref(A_MOV,S_L,0,r);            end;            hp:=hp^.next;         end;   end;  procedure finalizetempansistrings;    var       hp : ptemprecord;       hr : treference;    begin       hp:=templist;       while assigned(hp) do         begin            if hp^.temptype in [tt_ansistring,tt_freeansistring] then              begin                 procinfo^.flags:=procinfo^.flags or pi_needs_implicit_finally;                 reset_reference(hr);                 hr.base:=procinfo^.framepointer;                 hr.offset:=hp^.pos;                 emitpushreferenceaddr(hr);                 emitcall('FPC_ANSISTR_DECR_REF');              end;            hp:=hp^.next;         end;   end;  var     ls : longint;  procedure largest_size(p : pnamedindexobject);{$ifndef FPC}far;{$endif}    begin       if (psym(p)^.typ=varsym) and         (pvarsym(p)^.getvaluesize>ls) then         ls:=pvarsym(p)^.getvaluesize;    end;  procedure alignstack(alist : paasmoutput);    begin{$ifdef dummy}       if (cs_optimize in aktglobalswitches) and         (aktoptprocessor in [classp5,classp6]) then         begin            ls:=0;            aktprocsym^.definition^.localst^.foreach({$ifndef TP}@{$endif}largest_size);            if ls>=8 then              alist^.insert(new(paicpu,op_const_reg(A_AND,S_L,-8,R_ESP)));         end;{$endif dummy}    end;  procedure genentrycode(alist : paasmoutput;const proc_names:Tstringcontainer;make_global:boolean;                         stackframe:longint;                         var parasize:longint;var nostackframe:boolean;                         inlined : boolean);  {    Generates the entry code for a procedure  }    var      hs : string;{$ifdef GDB}      stab_function_name : Pai_stab_function_name;{$endif GDB}      hr : preference;      p : psymtable;      r : treference;      oldlist,      oldexprasmlist : paasmoutput;      again : pasmlabel;      i : longint;    begin       oldexprasmlist:=exprasmlist;       exprasmlist:=alist;       if (not inlined) and (aktprocsym^.definition^.proctypeoption=potype_proginit) then           begin              emitinsertcall('FPC_INITIALIZEUNITS');              if target_info.target=target_I386_WIN32 then                begin                   new(hr);                   reset_reference(hr^);                   hr^.symbol:=newasmsymbol(                   'U_SYSWIN32_ISCONSOLE');                   if apptype=at_cui then                     exprasmlist^.insert(new(paicpu,op_const_ref(A_MOV,S_B,                       1,hr)))                   else                     exprasmlist^.insert(new(paicpu,op_const_ref(A_MOV,S_B,                       0,hr)));                end;              oldlist:=exprasmlist;              exprasmlist:=new(paasmoutput,init);              p:=symtablestack;              while assigned(p) do                begin                   p^.foreach({$ifndef TP}@{$endif}initialize_threadvar);                   p:=p^.next;                end;              oldlist^.insertlist(exprasmlist);              dispose(exprasmlist,done);              exprasmlist:=oldlist;           end;{$ifdef GDB}      if (not inlined) and (cs_debuginfo in aktmoduleswitches) then        exprasmlist^.insert(new(pai_force_line,init));{$endif GDB}      { a constructor needs a help procedure }      if (aktprocsym^.definition^.proctypeoption=potype_constructor) then        begin          if procinfo^._class^.is_class then            begin              procinfo^.flags:=procinfo^.flags or pi_needs_implicit_finally;              exprasmlist^.insert(new(paicpu,op_cond_sym(A_Jcc,C_Z,S_NO,faillabel)));              emitinsertcall('FPC_NEW_CLASS');            end          else            begin              exprasmlist^.insert(new(paicpu,op_cond_sym(A_Jcc,C_Z,S_NO,faillabel)));              emitinsertcall('FPC_HELP_CONSTRUCTOR');{$ifndef noAllocEdi}              getexplicitregister32(R_EDI);{$endif noAllocEdi}              exprasmlist^.insert(new(paicpu,op_const_reg(A_MOV,S_L,procinfo^._class^.vmt_offset,R_EDI)));            end;        end;      { don't load ESI, does the caller }      { we must do it for local function }      { that can be called from a foreach }      { of another object than self !! PM }         if assigned(procinfo^._class) and            (lexlevel>normal_function_level) then           maybe_loadesi;      { When message method contains self as a parameter,        we must load it into ESI }      If (po_containsself in aktprocsym^.definition^.procoptions) then        begin           new(hr);           reset_reference(hr^);           hr^.offset:=procinfo^.selfpointer_offset;           hr^.base:=procinfo^.framepointer;           exprasmlist^.insert(new(paicpu,op_ref_reg(A_MOV,S_L,hr,R_ESI)));{$ifndef noAllocEdi}           exprasmlist^.insert(new(pairegalloc,alloc(R_ESI)));{$endif noAllocEdi}        end;      { should we save edi,esi,ebx like C ? }      if (po_savestdregs in aktprocsym^.definition^.procoptions) then       begin         if (aktprocsym^.definition^.usedregisters and ($80 shr byte(R_EBX)))<>0 then           exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_EBX)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_ESI)));         exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_EDI)));       end;      { for the save all registers we can simply use a pusha,popa which        push edi,esi,ebp,esp(ignored),ebx,edx,ecx,eax }      if (po_saveregisters in aktprocsym^.definition^.procoptions) then        begin          exprasmlist^.insert(new(paicpu,op_none(A_PUSHA,S_L)));        end;      { omit stack frame ? }      if not inlined then      if procinfo^.framepointer=stack_pointer then          begin              CGMessage(cg_d_stackframe_omited);              nostackframe:=true;              if (aktprocsym^.definition^.proctypeoption in [potype_unitinit,potype_proginit,potype_unitfinalize]) then                parasize:=0              else                parasize:=aktprocsym^.definition^.parast^.datasize+procinfo^.para_offset-4;              if stackframe<>0 then                exprasmlist^.insert(new(paicpu,                  op_const_reg(A_SUB,S_L,stackframe,R_ESP)));          end      else          begin              alignstack(alist);              if (aktprocsym^.definition^.proctypeoption in [potype_unitinit,potype_proginit,potype_unitfinalize]) then                parasize:=0              else                parasize:=aktprocsym^.definition^.parast^.datasize+procinfo^.para_offset-8;              nostackframe:=false;              if stackframe<>0 then                  begin{$ifdef unused}                      if (cs_littlesize in aktglobalswitches) and (stackframe<=65535) then                          begin                              if (cs_check_stack in aktlocalswitches) and                                 not(target_info.target in [target_i386_linux,target_i386_win32]) then                                begin                                  emitinsertcall('FPC_STACKCHECK');                                  exprasmlist^.insert(new(paicpu,op_const(A_PUSH,S_L,stackframe)));                                end;                              if cs_profile in aktmoduleswitches then                                genprofilecode;                            { %edi is already saved when pocdecl is used                              if (target_info.target=target_linux) and                               ((aktprocsym^.definition^.options and poexports)<>0) then                                  exprasmlist^.insert(new(Paicpu,op_reg(A_PUSH,S_L,R_EDI))); }                              { ATTENTION:                                never use ENTER in linux !!!                                the stack page fault does not support it PM }                              exprasmlist^.insert(new(paicpu,op_const_const(A_ENTER,S_NO,stackframe,0)))                          end                      else{$endif unused}                          begin                            { windows guards only a few pages for stack growing, }                            { so we have to access every page first              }                            if (target_os.id=os_i386_win32) and                              (stackframe>=winstackpagesize) then                              begin                                  if stackframe div winstackpagesize<=5 then                                    begin                                       exprasmlist^.insert(new(paicpu,op_const_reg(A_SUB,S_L,stackframe-4,R_ESP)));                                       for i:=1 to stackframe div winstackpagesize do                                         begin                                            hr:=new_reference(R_ESP,stackframe-i*winstackpagesize);                                            exprasmlist^.concat(new(paicpu,                                              op_const_ref(A_MOV,S_L,0,hr)));                                         end;                                       exprasmlist^.concat(new(paicpu,                                         op_reg(A_PUSH,S_L,R_EAX)));                                    end                                  else                                    begin                                       getlabel(again);{$ifndef noAllocEdi}                                       getexplicitregister32(R_EDI);{$endif noAllocEdi}                                       exprasmlist^.concat(new(paicpu,                                         op_const_reg(A_MOV,S_L,stackframe div winstackpagesize,R_EDI)));                                       emitlab(again);                                       exprasmlist^.concat(new(paicpu,                                         op_const_reg(A_SUB,S_L,winstackpagesize-4,R_ESP)));                                       exprasmlist^.concat(new(paicpu,                                         op_reg(A_PUSH,S_L,R_EAX)));                                       exprasmlist^.concat(new(paicpu,                                         op_reg(A_DEC,S_L,R_EDI)));                                       emitjmp(C_NZ,again);{$ifndef noAllocEdi}                                       ungetregister32(R_EDI);{$endif noAllocEdi}                                       exprasmlist^.concat(new(paicpu,                                         op_const_reg(A_SUB,S_L,stackframe mod winstackpagesize,R_ESP)));                                    end                              end                            else                              exprasmlist^.insert(new(paicpu,op_const_reg(A_SUB,S_L,stackframe,R_ESP)));                            if (cs_check_stack in aktlocalswitches) and                              not(target_info.target in [target_i386_linux,target_i386_win32]) then                              begin                                 emitinsertcall('FPC_STACKCHECK');                                 exprasmlist^.insert(new(paicpu,op_const(A_PUSH,S_L,stackframe)));                              end;                            if cs_profile in aktmoduleswitches then                              genprofilecode;                            exprasmlist^.insert(new(paicpu,op_reg_reg(A_MOV,S_L,R_ESP,R_EBP)));                            exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_EBP)));                          end;                  end { endif stackframe <> 0 }              else                 begin                   if cs_profile in aktmoduleswitches then                     genprofilecode;                   exprasmlist^.insert(new(paicpu,op_reg_reg(A_MOV,S_L,R_ESP,R_EBP)));                   exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_EBP)));                 end;          end;      if (po_interrupt in aktprocsym^.definition^.procoptions) then          generate_interrupt_stackframe_entry;      { initialize return value }      if (procinfo^.returntype.def<>pdef(voiddef)) and        (procinfo^.returntype.def^.needs_inittable) and        ((procinfo^.returntype.def^.deftype<>objectdef) or        not(pobjectdef(procinfo^.returntype.def)^.is_class)) then        begin           procinfo^.flags:=procinfo^.flags or pi_needs_implicit_finally;           reset_reference(r);           r.offset:=procinfo^.return_offset;           r.base:=procinfo^.framepointer;           initialize(procinfo^.returntype.def,r,ret_in_param(procinfo^.returntype.def));        end;      { generate copies of call by value parameters }      if not(po_assembler in aktprocsym^.definition^.procoptions) and         not (pocall_cdecl in aktprocsym^.definition^.proccalloptions) then        aktprocsym^.definition^.parast^.foreach({$ifndef TP}@{$endif}copyvalueparas);      { initialisize local data like ansistrings }      case aktprocsym^.definition^.proctypeoption of         potype_unitinit:           begin              { using current_module^.globalsymtable is hopefully      }              { more robust than symtablestack and symtablestack^.next }              psymtable(current_module^.globalsymtable)^.foreach({$ifndef TP}@{$endif}initialize_data);              psymtable(current_module^.localsymtable)^.foreach({$ifndef TP}@{$endif}initialize_data);           end;         { units have seperate code for initilization and finalization }         potype_unitfinalize: ;         else           aktprocsym^.definition^.localst^.foreach({$ifndef TP}@{$endif}initialize_data);      end;      { add a reference to all call by value/const parameters }      aktprocsym^.definition^.parast^.foreach({$ifndef TP}@{$endif}incr_data);      { initialisizes temp. ansi/wide string data }      inittempansistrings;      { do we need an exception frame because of ansi/widestrings ? }      if not inlined and         ((procinfo^.flags and pi_needs_implicit_finally)<>0) and      { but it's useless in init/final code of units }        not(aktprocsym^.definition^.proctypeoption in [potype_unitfinalize,potype_unitinit]) then        begin            usedinproc:=usedinproc or ($80 shr byte(R_EAX));            { Type of stack-frame must be pushed}            exprasmlist^.concat(new(paicpu,op_const(A_PUSH,S_L,1)));            emitcall('FPC_PUSHEXCEPTADDR');            exprasmlist^.concat(new(paicpu,              op_reg(A_PUSH,S_L,R_EAX)));            emitcall('FPC_SETJMP');            exprasmlist^.concat(new(paicpu,              op_reg(A_PUSH,S_L,R_EAX)));            exprasmlist^.concat(new(paicpu,              op_reg_reg(A_TEST,S_L,R_EAX,R_EAX)));            emitjmp(C_NE,aktexitlabel);            { probably we've to reload self here }            maybe_loadesi;        end;      if not inlined then       begin         if (cs_profile in aktmoduleswitches) or            (aktprocsym^.definition^.owner^.symtabletype=globalsymtable) or            (assigned(procinfo^._class) and (procinfo^._class^.owner^.symtabletype=globalsymtable)) then              make_global:=true;         hs:=proc_names.get;{$ifdef GDB}         if (cs_debuginfo in aktmoduleswitches) and target_os.use_function_relative_addresses then           stab_function_name := new(pai_stab_function_name,init(strpnew(hs)));{$EndIf GDB}         while hs<>'' do          begin            if make_global then              exprasmlist^.insert(new(pai_symbol,initname_global(hs,0)))            else              exprasmlist^.insert(new(pai_symbol,initname(hs,0)));{$ifdef GDB}            if (cs_debuginfo in aktmoduleswitches) and               target_os.use_function_relative_addresses then              exprasmlist^.insert(new(pai_stab_function_name,init(strpnew(hs))));{$endif GDB}            hs:=proc_names.get;          end;         if make_global or ((procinfo^.flags and pi_is_global) <> 0) then          aktprocsym^.is_global := True;{$ifdef GDB}         if (cs_debuginfo in aktmoduleswitches) then          begin            if target_os.use_function_relative_addresses then             exprasmlist^.insert(stab_function_name);            exprasmlist^.insert(new(pai_stabs,init(aktprocsym^.stabstring)));            aktprocsym^.isstabwritten:=true;          end;{$endif GDB}       { Align, gprof uses 16 byte granularity }         if (cs_profile in aktmoduleswitches) then          exprasmlist^.insert(new(pai_align,init_op(16,$90)))         else          if not(cs_littlesize in aktglobalswitches) then           exprasmlist^.insert(new(pai_align,init(16)));       end;      exprasmlist:=oldexprasmlist;  end;  procedure handle_return_value(inlined : boolean;var uses_eax,uses_edx : boolean);    var       hr : preference;       op : Tasmop;       s : Topsize;  begin      uses_eax:=false;      uses_edx:=false;      if procinfo^.returntype.def<>pdef(voiddef) then          begin              {if ((procinfo^.flags and pi_operator)<>0) and                 assigned(opsym) then                procinfo^.funcret_is_valid:=                  procinfo^.funcret_is_valid or (opsym^.refs>0);}              if (procinfo^.funcret_state<>vs_assigned) and not inlined { and                ((procinfo^.flags and pi_uses_asm)=0)} then               CGMessage(sym_w_function_result_not_set);              hr:=new_reference(procinfo^.framepointer,procinfo^.return_offset);              if (procinfo^.returntype.def^.deftype in [orddef,enumdef]) then                begin                  uses_eax:=true;                  case procinfo^.returntype.def^.size of                   8:                     begin                        emit_ref_reg(A_MOV,S_L,hr,R_EAX);                        hr:=new_reference(procinfo^.framepointer,procinfo^.return_offset+4);                        emit_ref_reg(A_MOV,S_L,hr,R_EDX);                        uses_edx:=true;                     end;                   4:                     emit_ref_reg(A_MOV,S_L,hr,R_EAX);                   2:                     emit_ref_reg(A_MOV,S_W,hr,R_AX);                   1:                     emit_ref_reg(A_MOV,S_B,hr,R_AL);                  end;                end              else                if ret_in_acc(procinfo^.returntype.def) then                  begin                    uses_eax:=true;                    emit_ref_reg(A_MOV,S_L,hr,R_EAX);                  end              else                 if (procinfo^.returntype.def^.deftype=floatdef) then                   begin                      floatloadops(pfloatdef(procinfo^.returntype.def)^.typ,op,s);                      exprasmlist^.concat(new(paicpu,op_ref(op,s,hr)))                   end              else                dispose(hr);          end  end;  procedure genexitcode(alist : paasmoutput;parasize:longint;nostackframe,inlined:boolean);    var{$ifdef GDB}       mangled_length : longint;       p : pchar;{$endif GDB}       nofinal,okexitlabel,noreraiselabel,nodestroycall : pasmlabel;       hr : treference;       uses_eax,uses_edx,uses_esi : boolean;       oldexprasmlist : paasmoutput;       ai : paicpu;       pd : pprocdef;  begin      oldexprasmlist:=exprasmlist;      exprasmlist:=alist;      if aktexitlabel^.is_used then        exprasmlist^.insert(new(pai_label,init(aktexitlabel)));      { call the destructor help procedure }      if (aktprocsym^.definition^.proctypeoption=potype_destructor) and         assigned(procinfo^._class) then        begin          if procinfo^._class^.is_class then            begin              emitinsertcall('FPC_DISPOSE_CLASS');            end          else            begin              emitinsertcall('FPC_HELP_DESTRUCTOR');{$ifndef noAllocEdi}              getexplicitregister32(R_EDI);{$endif noAllocEdi}              exprasmlist^.insert(new(paicpu,op_const_reg(A_MOV,S_L,procinfo^._class^.vmt_offset,R_EDI)));              { must the object be finalized ? }              if procinfo^._class^.needs_inittable then                begin                   getlabel(nofinal);                   exprasmlist^.insert(new(pai_label,init(nofinal)));                   emitinsertcall('FPC_FINALIZE');{$ifndef noAllocEdi}                   ungetregister32(R_EDI);{$endif noAllocEdi}                   exprasmlist^.insert(new(paicpu,op_reg(A_PUSH,S_L,R_ESI)));                   exprasmlist^.insert(new(paicpu,op_sym(A_PUSH,S_L,procinfo^._class^.get_inittable_label)));                   ai:=new(paicpu,op_sym(A_Jcc,S_NO,nofinal));                   ai^.SetCondition(C_Z);                   exprasmlist^.insert(ai);                   reset_reference(hr);                   hr.base:=R_EBP;                   hr.offset:=8;                   exprasmlist^.insert(new(paicpu,op_const_ref(A_CMP,S_L,0,newreference(hr))));                end;            end;        end;      { finalize temporary data }      finalizetempansistrings;      { finalize local data like ansistrings}      case aktprocsym^.definition^.proctypeoption of         potype_unitfinalize:           begin              { using current_module^.globalsymtable is hopefully      }              { more robust than symtablestack and symtablestack^.next }              psymtable(current_module^.globalsymtable)^.foreach({$ifndef TP}@{$endif}finalize_data);              psymtable(current_module^.localsymtable)^.foreach({$ifndef TP}@{$endif}finalize_data);           end;         { units have seperate code for initialization and finalization }         potype_unitinit: ;         else           aktprocsym^.definition^.localst^.foreach({$ifndef TP}@{$endif}finalize_data);      end;      { finalize paras data }      if assigned(aktprocsym^.definition^.parast) then        aktprocsym^.definition^.parast^.foreach({$ifndef TP}@{$endif}finalize_data);      { do we need to handle exceptions because of ansi/widestrings ? }      if not inlined and         ((procinfo^.flags and pi_needs_implicit_finally)<>0) and      { but it's useless in init/final code of units }        not(aktprocsym^.definition^.proctypeoption in [potype_unitfinalize,potype_unitinit]) then        begin           { the exception helper routines modify all registers }           aktprocsym^.definition^.usedregisters:=$ff;           getlabel(noreraiselabel);           emitcall('FPC_POPADDRSTACK');           exprasmlist^.concat(new(paicpu,             op_reg(A_POP,S_L,R_EAX)));           exprasmlist^.concat(new(paicpu,             op_reg_reg(A_TEST,S_L,R_EAX,R_EAX)));           emitjmp(C_E,noreraiselabel);           if (aktprocsym^.definition^.proctypeoption=potype_constructor) then             begin                if assigned(procinfo^._class) then                  begin                     pd:=procinfo^._class^.searchdestructor;                     if assigned(pd) then                       begin                          getlabel(nodestroycall);                          emit_const_ref(A_CMP,S_L,0,new_reference(procinfo^.framepointer,                            procinfo^.selfpointer_offset));                          emitjmp(C_E,nodestroycall);                          if procinfo^._class^.is_class then                            begin                               emit_const(A_PUSH,S_L,1);                               emit_reg(A_PUSH,S_L,R_ESI);                            end                          else                            begin                               emit_reg(A_PUSH,S_L,R_ESI);                               emit_sym(A_PUSH,S_L,newasmsymbol(procinfo^._class^.vmt_mangledname));                            end;                          if (po_virtualmethod in pd^.procoptions) then                            begin                               emit_ref_reg(A_MOV,S_L,new_reference(R_ESI,0),R_EDI);                               emit_ref(A_CALL,S_NO,new_reference(R_EDI,procinfo^._class^.vmtmethodoffset(pd^.extnumber)));                            end                          else                            emitcall(pd^.mangledname);                          { not necessary because the result is never assigned in the                            case of an exception (FK)                          emit_const_reg(A_MOV,S_L,0,R_ESI);                          emit_const_ref(A_MOV,S_L,0,new_reference(procinfo^.framepointer,8));                          }                          emitlab(nodestroycall);                       end;                  end             end           else           { must be the return value finalized before reraising the exception? }           if (procinfo^.returntype.def<>pdef(voiddef)) and             (procinfo^.returntype.def^.needs_inittable) and             ((procinfo^.returntype.def^.deftype<>objectdef) or             not(pobjectdef(procinfo^.returntype.def)^.is_class)) then             begin                reset_reference(hr);                hr.offset:=procinfo^.return_offset;                hr.base:=procinfo^.framepointer;                finalize(procinfo^.returntype.def,hr,ret_in_param(procinfo^.returntype.def));             end;           emitcall('FPC_RERAISE');           emitlab(noreraiselabel);        end;      { call __EXIT for main program }      if (not DLLsource) and (not inlined) and (aktprocsym^.definition^.proctypeoption=potype_proginit) then       begin         emitcall('FPC_DO_EXIT');       end;      { handle return value }      uses_eax:=false;      uses_edx:=false;      uses_esi:=false;      if not(po_assembler in aktprocsym^.definition^.procoptions) then          if (aktprocsym^.definition^.proctypeoption<>potype_constructor) then            handle_return_value(inlined,uses_eax,uses_edx)          else              begin                  { successful constructor deletes the zero flag }                  { and returns self in eax                   }                  { eax must be set to zero if the allocation failed !!! }                  getlabel(okexitlabel);                  emitjmp(C_NONE,okexitlabel);                  emitlab(faillabel);                  if procinfo^._class^.is_class then                    begin                      emit_ref_reg(A_MOV,S_L,new_reference(procinfo^.framepointer,8),R_ESI);                      emitcall('FPC_HELP_FAIL_CLASS');                    end                  else                    begin                      emit_ref_reg(A_MOV,S_L,new_reference(procinfo^.framepointer,12),R_ESI);{$ifndef noAllocEdi}                       getexplicitregister32(R_EDI);{$endif noAllocEdi}                      emit_const_reg(A_MOV,S_L,procinfo^._class^.vmt_offset,R_EDI);                      emitcall('FPC_HELP_FAIL');{$ifndef noAllocEdi}                      ungetregister32(R_EDI);{$endif noAllocEdi}                    end;                  emitlab(okexitlabel);                  emit_reg_reg(A_MOV,S_L,R_ESI,R_EAX);                  emit_reg_reg(A_TEST,S_L,R_ESI,R_ESI);                  uses_eax:=true;                  uses_esi:=true;              end;      { stabs uses the label also ! }      if aktexit2label^.is_used or         ((cs_debuginfo in aktmoduleswitches) and not inlined) then        emitlab(aktexit2label);      { gives problems for long mangled names }      {list^.concat(new(pai_symbol,init(aktprocsym^.definition^.mangledname+'_end')));}      { should we restore edi ? }      { for all i386 gcc implementations }      if (po_savestdregs in aktprocsym^.definition^.procoptions) then        begin          if (aktprocsym^.definition^.usedregisters and ($80 shr byte(R_EBX)))<>0 then           exprasmlist^.concat(new(paicpu,op_reg(A_POP,S_L,R_EBX)));          exprasmlist^.concat(new(paicpu,op_reg(A_POP,S_L,R_ESI)));          exprasmlist^.concat(new(paicpu,op_reg(A_POP,S_L,R_EDI)));          { here we could reset R_EBX            but that is risky because it only works            if genexitcode is called after genentrycode            so lets skip this for the moment PM          aktprocsym^.definition^.usedregisters:=            aktprocsym^.definition^.usedregisters or not ($80 shr byte(R_EBX));          }        end;      { for the save all registers we can simply use a pusha,popa which        push edi,esi,ebp,esp(ignored),ebx,edx,ecx,eax }      if (po_saveregisters in aktprocsym^.definition^.procoptions) then        begin          if uses_esi then            exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_ESI,new_reference(R_ESP,4))));          if uses_edx then            exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDX,new_reference(R_ESP,20))));          if uses_eax then            exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EAX,new_reference(R_ESP,28))));          exprasmlist^.concat(new(paicpu,op_none(A_POPA,S_L)))        end;      if not(nostackframe) then        begin          if not inlined then            exprasmlist^.concat(new(paicpu,op_none(A_LEAVE,S_NO)));        end      else        begin          if (gettempsize<>0) and not inlined then            exprasmlist^.insert(new(paicpu,              op_const_reg(A_ADD,S_L,gettempsize,R_ESP)));        end;      { parameters are limited to 65535 bytes because }      { ret allows only imm16                    }      if (parasize>65535) and not(pocall_clearstack in aktprocsym^.definition^.proccalloptions) then       CGMessage(cg_e_parasize_too_big);      { at last, the return is generated }      if not inlined then      if (po_interrupt in aktprocsym^.definition^.procoptions) then          begin             if uses_esi then               exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_ESI,new_reference(R_ESP,16))));             if uses_edx then               exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EDX,new_reference(R_ESP,12))));             if uses_eax then               exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,S_L,R_EAX,new_reference(R_ESP,0))));             generate_interrupt_stackframe_exit;          end      else       begin       {Routines with the poclearstack flag set use only a ret.}       { also routines with parasize=0     }         if (pocall_clearstack in aktprocsym^.definition^.proccalloptions) then           begin{$ifndef OLD_C_STACK}             { complex return values are removed from stack in C code PM }             if ret_in_param(aktprocsym^.definition^.rettype.def) then               exprasmlist^.concat(new(paicpu,op_const(A_RET,S_NO,4)))             else{$endif not OLD_C_STACK}               exprasmlist^.concat(new(paicpu,op_none(A_RET,S_NO)));           end         else if (parasize=0) then          exprasmlist^.concat(new(paicpu,op_none(A_RET,S_NO)))         else          exprasmlist^.concat(new(paicpu,op_const(A_RET,S_NO,parasize)));       end;      if not inlined then        exprasmlist^.concat(new(pai_symbol_end,initname(aktprocsym^.definition^.mangledname)));{$ifdef GDB}      if (cs_debuginfo in aktmoduleswitches) and not inlined  then          begin              aktprocsym^.concatstabto(exprasmlist);              if assigned(procinfo^._class) then                if (not assigned(procinfo^.parent) or                   not assigned(procinfo^.parent^._class)) then                  exprasmlist^.concat(new(pai_stabs,init(strpnew(                   '"$t:v'+procinfo^._class^.numberstring+'",'+                   tostr(N_PSYM)+',0,0,'+tostr(procinfo^.selfpointer_offset)))))                else                  exprasmlist^.concat(new(pai_stabs,init(strpnew(                   '"$t:r*'+procinfo^._class^.numberstring+'",'+                   tostr(N_RSYM)+',0,0,'+tostr(GDB_i386index[R_ESI])))));              { define calling EBP as pseudo local var PM }              { this enables test if the function is a local one !! }              if  assigned(procinfo^.parent) and (lexlevel>normal_function_level) then                exprasmlist^.concat(new(pai_stabs,init(strpnew(                 '"parent_ebp:'+voidpointerdef^.numberstring+'",'+                 tostr(N_LSYM)+',0,0,'+tostr(procinfo^.framepointer_offset)))));              if (pdef(aktprocsym^.definition^.rettype.def) <> pdef(voiddef)) then                begin                  if ret_in_param(aktprocsym^.definition^.rettype.def) then                    exprasmlist^.concat(new(pai_stabs,init(strpnew(                     '"'+aktprocsym^.name+':X*'+aktprocsym^.definition^.rettype.def^.numberstring+'",'+                     tostr(N_PSYM)+',0,0,'+tostr(procinfo^.return_offset)))))                  else                    exprasmlist^.concat(new(pai_stabs,init(strpnew(                     '"'+aktprocsym^.name+':X'+aktprocsym^.definition^.rettype.def^.numberstring+'",'+                     tostr(N_PSYM)+',0,0,'+tostr(procinfo^.return_offset)))));                  if (m_result in aktmodeswitches) then                    if ret_in_param(aktprocsym^.definition^.rettype.def) then                      exprasmlist^.concat(new(pai_stabs,init(strpnew(                       '"RESULT:X*'+aktprocsym^.definition^.rettype.def^.numberstring+'",'+                       tostr(N_PSYM)+',0,0,'+tostr(procinfo^.return_offset)))))                    else                      exprasmlist^.concat(new(pai_stabs,init(strpnew(                       '"RESULT:X'+aktprocsym^.definition^.rettype.def^.numberstring+'",'+                       tostr(N_PSYM)+',0,0,'+tostr(procinfo^.return_offset)))));                end;              mangled_length:=length(aktprocsym^.definition^.mangledname);              getmem(p,2*mangled_length+50);              strpcopy(p,'192,0,0,');              strpcopy(strend(p),aktprocsym^.definition^.mangledname);              if (target_os.use_function_relative_addresses) then                begin                  strpcopy(strend(p),'-');                  strpcopy(strend(p),aktprocsym^.definition^.mangledname);                end;              exprasmlist^.concat(new(pai_stabn,init(strnew(p))));              {list^.concat(new(pai_stabn,init(strpnew('192,0,0,'               +aktprocsym^.definition^.mangledname))));              p[0]:='2';p[1]:='2';p[2]:='4';              strpcopy(strend(p),'_end');}              strpcopy(p,'224,0,0,'+aktexit2label^.name);              if (target_os.use_function_relative_addresses) then                begin                  strpcopy(strend(p),'-');                  strpcopy(strend(p),aktprocsym^.definition^.mangledname);                end;              exprasmlist^.concatlist(withdebuglist);              exprasmlist^.concat(new(pai_stabn,init(                strnew(p))));               { strpnew('224,0,0,'               +aktprocsym^.definition^.mangledname+'_end'))));}              freemem(p,2*mangled_length+50);          end;{$endif GDB}      exprasmlist:=oldexprasmlist;  end;    procedure genimplicitunitfinal(alist : paasmoutput);      begin         { using current_module^.globalsymtable is hopefully      }         { more robust than symtablestack and symtablestack^.next }         psymtable(current_module^.globalsymtable)^.foreach({$ifndef TP}@{$endif}finalize_data);         psymtable(current_module^.localsymtable)^.foreach({$ifndef TP}@{$endif}finalize_data);         exprasmlist^.insert(new(pai_symbol,initname_global('FINALIZE$$'+current_module^.modulename^,0)));         exprasmlist^.insert(new(pai_symbol,initname_global(target_os.cprefix+current_module^.modulename^+'_finalize',0)));{$ifdef GDB}         if (cs_debuginfo in aktmoduleswitches) and           target_os.use_function_relative_addresses then           exprasmlist^.insert(new(pai_stab_function_name,init(strpnew('FINALIZE$$'+current_module^.modulename^))));{$endif GDB}         exprasmlist^.concat(new(paicpu,op_none(A_RET,S_NO)));         alist^.concatlist(exprasmlist);      end;    procedure genimplicitunitinit(alist : paasmoutput);      begin         { using current_module^.globalsymtable is hopefully      }         { more robust than symtablestack and symtablestack^.next }         psymtable(current_module^.globalsymtable)^.foreach({$ifndef TP}@{$endif}finalize_data);         psymtable(current_module^.localsymtable)^.foreach({$ifndef TP}@{$endif}finalize_data);         exprasmlist^.insert(new(pai_symbol,initname_global('INIT$$'+current_module^.modulename^,0)));         exprasmlist^.insert(new(pai_symbol,initname_global(target_os.cprefix+current_module^.modulename^+'_init',0)));{$ifdef GDB}         if (cs_debuginfo in aktmoduleswitches) and           target_os.use_function_relative_addresses then           exprasmlist^.insert(new(pai_stab_function_name,init(strpnew('INIT$$'+current_module^.modulename^))));{$endif GDB}         exprasmlist^.concat(new(paicpu,op_none(A_RET,S_NO)));         alist^.concatlist(exprasmlist);      end;{$ifdef test_dest_loc}       procedure mov_reg_to_dest(p : ptree; s : topsize; reg : tregister);         begin            if (dest_loc.loc=LOC_CREGISTER) or (dest_loc.loc=LOC_REGISTER) then              begin                emit_reg_reg(A_MOV,s,reg,dest_loc.register);                set_location(p^.location,dest_loc);                in_dest_loc:=true;              end            else            if (dest_loc.loc=LOC_REFERENCE) or (dest_loc.loc=LOC_MEM) then              begin                exprasmlist^.concat(new(paicpu,op_reg_ref(A_MOV,s,reg,newreference(dest_loc.reference))));                set_location(p^.location,dest_loc);                in_dest_loc:=true;              end            else              internalerror(20080);         end;{$endif test_dest_loc}end.{  $Log$  Revision 1.1  2000-07-13 06:29:47  michael  + Initial import  Revision 1.109  2000/06/27 12:17:29  jonas    * fix for web bug 1011: no exception stack stuff is generated for      inlined procedures, the entry/exitcode of the parent will do that  Revision 1.108  2000/06/10 17:31:42  jonas    * loadord2reg doesn't generate any "movl %reg1,%reg1" anymore  Revision 1.107  2000/06/05 20:39:05  pierre   * fix for inline bug  Revision 1.106  2000/05/26 20:16:00  jonas    * fixed wrong register deallocations in several ansistring related      procedures. The IDE's now function fine when compiled with -OG3p3r  Revision 1.105  2000/05/23 14:20:49  pierre   * Use stacksize param instead of gettempsize  Revision 1.104  2000/05/18 17:05:15  peter    * fixed size of const parameters in asm readers  Revision 1.103  2000/05/17 11:06:11  pierre   add a comment about ENTER and linux  Revision 1.102  2000/05/14 18:49:04  florian    + Int64/QWord stuff for array of const added  Revision 1.101  2000/05/09 14:17:33  pierre   * handle interrupt function correctly  Revision 1.100  2000/05/04 09:29:31  pierre   * saveregisters now does not overwrite registers used as return value for functions  Revision 1.99  2000/04/28 08:53:47  pierre   * fix my last fix for other targets then win32  Revision 1.98  2000/04/26 10:03:45  pierre    * correct bugs for ts010026 and ts010029 in win32 mode      in copyvaluparas    + use SHL instead of IMUL if constant is a power of 2 in copyvalueparas  Revision 1.97  2000/04/24 12:48:37  peter    * removed unused vars  Revision 1.96  2000/04/10 12:23:18  jonas    * modified copyshortstring so it takes an extra paramter which allows it      to delete the sref itself (so the reg deallocations are put in the      right place for the optimizer)  Revision 1.95  2000/04/10 09:01:15  pierre   * fix for bug 922 in copyvalueparas  Revision 1.94  2000/04/03 20:51:22  florian    * initialize/finalize_data checks if procinfo is assigned else      crashes happend at end of compiling if there were ansistrings in the      interface/implementation part of units: it was the result of the fix      of 701 :(  Revision 1.93  2000/04/02 10:18:18  florian    * bug 701 fixed: ansistrings in interface and implementation part of the units      are now finalized correctly even if there are no explicit initialization/      finalization statements  Revision 1.92  2000/04/01 14:18:45  peter    * use arraydef.elesize instead of elementtype.def.size  Revision 1.91  2000/03/31 22:56:46  pierre    * fix the handling of value parameters in cdecl function  Revision 1.90  2000/03/28 22:31:46  pierre   * fix for problem in tbs0299 for 4 byte stack alignment  Revision 1.89  2000/03/21 23:36:46  pierre   fix for bug 312  Revision 1.88  2000/03/19 11:55:08  peter    * fixed temp ansi handling within array constructor  Revision 1.87  2000/03/19 08:17:36  peter    * tp7 fix  Revision 1.86  2000/03/01 15:36:11  florian    * some new stuff for the new cg  Revision 1.85  2000/03/01 12:35:44  pierre   * fix for bug 855  Revision 1.84  2000/03/01 00:03:12  pierre    * fixes for locals in inlined procedures      fix for bug797    + stabs generation for inlined paras and locals  Revision 1.83  2000/02/18 21:25:48  florian    * fixed a bug in int64/qword handling was a quite ugly one  Revision 1.82  2000/02/18 20:53:14  pierre    * fixes a stabs problem for functions    + includes a stabs local var for with statements      the name is with in lowercase followed by an index      for nested with.    + Withdebuglist added because the stabs declarations of local      var are postponed to end of function.  Revision 1.81  2000/02/10 23:44:43  florian    * big update for exception handling code generation: possible mem holes      fixed, break/continue/exit should work always now as expected  Revision 1.80  2000/02/09 17:36:10  jonas    * added missing regalloc for ecx in range check code  Revision 1.79  2000/02/09 13:22:50  peter    * log truncated  Revision 1.78  2000/02/04 21:00:31  florian    * some (small) problems with register saving fixed  Revision 1.77  2000/02/04 20:00:21  florian    * an exception in a construcor calls now the destructor (this applies only      to classes)  Revision 1.76  2000/02/04 14:29:57  pierre   + add pseudo local var parent_ebp for local procs  Revision 1.75  2000/01/25 08:46:03  pierre   * Range check for int64 produces a warning only  Revision 1.74  2000/01/24 12:17:22  florian    * some improvemenst to cmov support    * disabled excpetion frame generation in cosntructors temporarily  Revision 1.73  2000/01/23 21:29:14  florian    * CMOV support in optimizer (in define USECMOV)    + start of support of exceptions in constructors  Revision 1.72  2000/01/23 11:11:36  michael  + Fixes from Jonas.  Revision 1.71  2000/01/22 16:02:37  jonas    * fixed more regalloc bugs (for set adding and unsigned      multiplication)  Revision 1.70  2000/01/16 22:17:11  peter    * renamed call_offset to para_offset  Revision 1.69  2000/01/12 10:38:17  peter    * smartlinking fixes for binary writer    * release alignreg code and moved instruction writing align to cpuasm,      but it doesn't use the specified register yet  Revision 1.68  2000/01/09 12:35:02  jonas    * changed edi allocation to use getexplicitregister32/ungetregister      (adapted tgeni386 a bit for this) and enabled it by default    * fixed very big and stupid bug of mine in cg386mat that broke the      include() code (and make cycle :( ) if you compiled without      -dnewoptimizations  Revision 1.67  2000/01/09 01:44:21  jonas    + (de)allocation info for EDI to fix reported bug on mailinglist.      Also some (de)allocation info for ESI added. Between -dallocEDI      because at this time of the night bugs could easily slip in ;)  Revision 1.66  2000/01/07 01:14:22  peter    * updated copyright to 2000  Revision 1.65  1999/12/22 01:01:47  peter    - removed freelabel()    * added undefined label detection in internal assembler, this prevents      a lot of ld crashes and wrong .o files    * .o files aren't written anymore if errors have occured    * inlining of assembler labels is now correct  Revision 1.64  1999/12/20 21:42:35  pierre    + dllversion global variable    * FPC_USE_CPREFIX code removed, not necessary anymore      as we use .edata direct writing by default now.  Revision 1.63  1999/12/01 22:45:54  peter    * fixed wrong assembler with in-node  Revision 1.62  1999/11/30 10:40:43  peter    + ttype, tsymlist  Revision 1.61  1999/11/20 01:22:18  pierre    + cond FPC_USE_CPREFIX (needs also some RTL changes)      this allows to use unit global vars as DLL exports      (the underline prefix seems needed by dlltool)  Revision 1.60  1999/11/17 17:04:58  pierre   * Notes/hints changes  Revision 1.59  1999/11/15 14:04:00  pierre   * self pointer stabs for local function was wrong}
 |