| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400 | {    Symbol table implementation for the definitions    Copyright (c) 1998-2005 by Florian Klaempfl, Pierre Muller    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 symdef;{$i fpcdefs.inc}interface    uses       { common }       cclasses,widestr,       { global }       globtype,globals,tokens,constexp,       { symtable }       symconst,symbase,symtype,       { node }       node,       { aasm }       aasmtai,       cpuinfo,       cgbase,       parabase       ;    type{************************************************                    TDef************************************************}       tgenericconstraintdata=class         interfaces : tfpobjectlist;         interfacesderef : tfplist;         flags : tgenericconstraintflags;         constructor create;         destructor destroy;override;         procedure ppuload(ppufile:tcompilerppufile);         procedure ppuwrite(ppufile:tcompilerppufile);         procedure buildderef;         procedure deref;       end;       { tstoreddef }       tstoreddef = class(tdef)       private{$ifdef symansistr}          _fullownerhierarchyname : ansistring;{$else symansistr}          _fullownerhierarchyname : pshortstring;{$endif symansistr}          procedure writeentry(ppufile: tcompilerppufile; ibnr: byte);       protected          typesymderef  : tderef;          procedure ppuwrite_platform(ppufile:tcompilerppufile);virtual;          procedure ppuload_platform(ppufile:tcompilerppufile);virtual;          { a (possibly) reusable def is always created on the basis of another            def, and contains a reference to this other def. If this other            def is in a non-persistent symboltable, the new def cannot actually            be safely reused everywhere in the current module. This routine            abtracts that checking, and also restores the symtable stack            (which had to be reset before creating the new def, so that the new             def did not automatically get added to its top) }          class procedure setup_reusable_def(origdef, newdef: tdef; res: PHashSetItem; oldsymtablestack: tsymtablestack);       public{$ifdef EXTDEBUG}          fileinfo   : tfileposinfo;{$endif}          { generic support }          genericdef      : tstoreddef;          genericdefderef : tderef;          generictokenbuf : tdynamicarray;          { this list contains references to the symbols that make up the            generic parameters; the symbols are not owned by this list            Note: this list is allocated on demand! }          genericparas    : tfphashobjectlist;          genericparaderefs : tfplist;          { contains additional data if this def is a generic constraint            Note: this class is allocated on demand! }          genconstraintdata : tgenericconstraintdata;          constructor create(dt:tdeftyp;doregister:boolean);          constructor ppuload(dt:tdeftyp;ppufile:tcompilerppufile);          destructor  destroy;override;          function getcopy : tstoreddef;virtual;          procedure ppuwrite(ppufile:tcompilerppufile);virtual;          procedure buildderef;override;          procedure buildderefimpl;override;          procedure deref;override;          procedure derefimpl;override;          function  size:asizeint;override;          function  getvardef:longint;override;          function  alignment:shortint;override;          function  is_publishable : boolean;override;          function  needs_inittable : boolean;override;          function  has_non_trivial_init_child(check_parent:boolean):boolean;override;          function  rtti_mangledname(rt:trttitype):TSymStr;override;          function  OwnerHierarchyName: string; override;          function  fullownerhierarchyname(skipprocparams:boolean):TSymStr;override;          function  needs_separate_initrtti:boolean;override;          function  in_currentunit: boolean;          { regvars }          function is_intregable : boolean;          function is_fpuregable : boolean;          { def can be put into a register if it is const/immutable }          function is_const_intregable : boolean;          { generics }          procedure initgeneric;          { this function can be used to determine whether a def is really a            generic declaration or just a normal type declared inside another            generic }          function is_generic:boolean;          { same as above for specializations }          function is_specialization:boolean;          { registers this def in the unit's deflist; no-op if already registered }          procedure register_def; override;          { add the def to the top of the symtable stack if it's not yet owned            by another symtable }          procedure maybe_put_in_symtable_stack;       private          savesize  : asizeuint;       end;       tfiletyp = (ft_text,ft_typed,ft_untyped);       tfiledef = class(tstoreddef)          filetyp : tfiletyp;          typedfiledef : tdef;          typedfiledefderef : tderef;          constructor createtext;virtual;          constructor createuntyped;virtual;          constructor createtyped(def : tdef);virtual;          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          procedure buildderef;override;          procedure deref;override;          function  GetTypeName:string;override;          function  getmangledparaname:TSymStr;override;          function  size:asizeint;override;          procedure setsize;          function alignment: shortint; override;       end;       tfiledefclass = class of tfiledef;       tvariantdef = class(tstoreddef)          varianttype : tvarianttype;          constructor create(v : tvarianttype);virtual;          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          function GetTypeName:string;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  getvardef:longint;override;          procedure setsize;          function is_publishable : boolean;override;          function needs_inittable : boolean;override;       end;       tvariantdefclass = class of tvariantdef;       tformaldef = class(tstoreddef)          typed:boolean;          constructor create(Atyped:boolean);virtual;          constructor ppuload(ppufile:tcompilerppufile);          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  GetTypeName:string;override;       end;       tformaldefclass = class of tformaldef;       tforwarddef = class(tstoreddef)          tosymname : pshortstring;          forwardpos : tfileposinfo;          constructor create(const s:string;const pos:tfileposinfo);virtual;          destructor destroy;override;          function getcopy:tstoreddef;override;          function GetTypeName:string;override;       end;       tforwarddefclass = class of tforwarddef;       tundefineddef = class(tstoreddef)          constructor create(doregister:boolean);virtual;          constructor ppuload(ppufile:tcompilerppufile);          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  GetTypeName:string;override;       end;       tundefineddefclass = class of tundefineddef;       terrordef = class(tstoreddef)          constructor create;virtual;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  GetTypeName:string;override;          function  getmangledparaname : TSymStr;override;       end;       terrordefclass = class of terrordef;       tabstractpointerdef = class(tstoreddef)          pointeddef : tdef;          pointeddefderef : tderef;          constructor create(dt:tdeftyp;def:tdef);          constructor ppuload(dt:tdeftyp;ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          function size:asizeint;override;          function alignment:shortint;override;       end;       tpointerdef = class(tabstractpointerdef)          has_pointer_math : boolean;          constructor create(def:tdef);virtual;          { returns a pointerdef for def, reusing an existing one in case it            exists in the current module }          class function getreusable(def: tdef): tpointerdef; virtual;          { same as above, but in case the def must never be freed after the            current module has been compiled -- even if the def was not written            to the ppu file (for defs in para locations, as we don't reset them            so we don't have to recalculate them all the time) }          class function getreusable_no_free(def: tdef): tpointerdef;          function size:asizeint;override;          function getcopy:tstoreddef;override;          constructor ppuload(ppufile:tcompilerppufile);          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  GetTypeName:string;override;          {# returns the appropriate int type for pointer arithmetic with the given pointer type.             When adding or subtracting a number to/from a pointer, this function returns the             int type to which that number has to be converted, before the operation can be performed.             Normally, this is sinttype, except on i8086, where it takes into account the             special i8086 pointer types (near, far, huge). }          function pointer_arithmetic_int_type:tdef;virtual;          {# the unsigned version of pointer_arithmetic_int_type. Used with inc/dec. }          function pointer_arithmetic_uint_type:tdef;virtual;          {# returns the int type produced when subtracting two pointers of the given type.             Normally, this is sinttype, except on i8086, where it takes into account the             special i8086 pointer types (near, far, huge). }          function pointer_subtraction_result_type:tdef;virtual;          function compatible_with_pointerdef_size(ptr: tpointerdef): boolean; virtual;          {# the integer index type used to convert the pointer to array (i.e. denotes int_type in: ptr[int_type]) }          function converted_pointer_to_array_range_type:tdef;virtual;       end;       tpointerdefclass = class of tpointerdef;       tprocdef = class;       tabstractrecorddef= class(tstoreddef)       private          rttistring     : string;       public          objname,          objrealname    : PShortString;          { for C++ classes: name of the library this class is imported from }          { for Java classes/records: package name }          import_lib     : PShortString;          symtable       : TSymtable;          cloneddef      : tabstractrecorddef;          cloneddefderef : tderef;          objectoptions  : tobjectoptions;          { for targets that initialise typed constants via explicit assignments            instead of by generating an initialised data sectino }          tcinitcode     : tnode;          constructor create(const n:string; dt:tdeftyp;doregister:boolean);          constructor ppuload(dt:tdeftyp;ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          destructor destroy; override;          procedure buildderefimpl;override;          procedure derefimpl;override;          procedure check_forwards; virtual;          function find_procdef_bytype(pt:tproctypeoption): tprocdef;          function GetSymtable(t:tGetSymtable):TSymtable;override;          function is_packed:boolean;          function RttiName: string;          { enumerator support }          function search_enumerator_get: tprocdef; virtual;          function search_enumerator_move: tprocdef; virtual;          function search_enumerator_current: tsym; virtual;          { JVM }          function jvm_full_typename(with_package_name: boolean): string;          { check if the symtable contains a float field }          function contains_float_field : boolean;          { check if the symtable contains a field that spans an aword boundary }          function contains_cross_aword_field: boolean;       end;       pvariantrecdesc = ^tvariantrecdesc;       tvariantrecbranch = record         { we store only single values here and no ranges because tvariantrecdesc is only needed in iso mode           which does not support range expressions in variant record definitions }         values : array of Tconstexprint;         nestedvariant : pvariantrecdesc;       end;       ppvariantrecdesc = ^pvariantrecdesc;       tvariantrecdesc = record         variantselector : tsym;         variantselectorderef : tderef;         branches : array of tvariantrecbranch;       end;       trecorddef = class(tabstractrecorddef)       public          variantrecdesc : pvariantrecdesc;          isunion       : boolean;          constructor create(const n:string; p:TSymtable);virtual;          constructor create_global_internal(n: string; packrecords, recordalignmin, maxCrecordalign: shortint); virtual;          function add_field_by_def(const optionalname: TIDString; def: tdef): tsym;          procedure add_fields_from_deflist(fieldtypes: tfplist);          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy;override;          function getcopy : tstoreddef;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          procedure buildderef;override;          procedure deref;override;          function  size:asizeint;override;          function  alignment : shortint;override;          function  padalignment: shortint;          function  GetTypeName:string;override;          { debug }          function  needs_inittable : boolean;override;          function  needs_separate_initrtti:boolean;override;          function  has_non_trivial_init_child(check_parent:boolean):boolean;override;       end;       trecorddefclass = class of trecorddef;       tobjectdef = class;       { TImplementedInterface }       TImplementedInterface = class        private         fIOffset      : longint;         function GetIOffset: longint;        public         IntfDef      : tobjectdef;         IntfDefDeref : tderef;         IType        : tinterfaceentrytype;         VtblImplIntf : TImplementedInterface;         NameMappings : TFPHashList;         ProcDefs     : TFPObjectList;         ImplementsGetter :  tsym;         ImplementsGetterDeref : tderef;         ImplementsField : tsym;         constructor create(aintf: tobjectdef);virtual;         constructor create_deref(intfd,getterd:tderef);virtual;         destructor  destroy; override;         function  getcopy:TImplementedInterface;         procedure buildderef;         procedure deref;         procedure AddMapping(const origname, newname: string);         function  GetMapping(const origname: string):string;         procedure AddImplProc(pd:tprocdef);         function  IsImplMergePossible(MergingIntf:TImplementedInterface;out weight: longint): boolean;         property  IOffset: longint read GetIOffset write fIOffset;       end;       timplementedinterfaceclass = class of timplementedinterface;       { tvmtentry }       tvmtentry = record         procdef      : tprocdef;         procdefderef : tderef;         visibility   : tvisibility;       end;       pvmtentry = ^tvmtentry;       { tobjectdef }       tvmcallstatic = (vmcs_default, vmcs_yes, vmcs_no, vmcs_unreachable);       pmvcallstaticinfo = ^tmvcallstaticinfo;       tmvcallstaticinfo = array[0..1024*1024-1] of tvmcallstatic;       tobjectdef = class(tabstractrecorddef)       private          fcurrent_dispid: longint;       public          childof        : tobjectdef;          childofderef   : tderef;          { for Object Pascal helpers }          extendeddef   : tdef;          extendeddefderef: tderef;          helpertype : thelpertype;          { for Objective-C: protocols and classes can have the same name there }          objextname     : pshortstring;          { to be able to have a variable vmt position }          { and no vmt field for objects without virtuals }          vmtentries     : TFPList;          vmcallstaticinfo : pmvcallstaticinfo;          vmt_field       : tsym;          vmt_fieldderef  : tderef;          iidguid        : pguid;          iidstr         : pshortstring;          { store implemented interfaces defs and name mappings }          ImplementedInterfaces : TFPObjectList;          { number of abstract methods (used by JVM target to determine whether            or not the class should be marked as abstract: must be done if 1 or            more abstract methods) }          abstractcnt    : longint;          writing_class_record_dbginfo,          { a class of this type has been created in this module }          created_in_current_module,          { a loadvmtnode for this class has been created in this            module, so if a classrefdef variable of this or a parent            class is used somewhere to instantiate a class, then this            class may be instantiated          }          maybe_created_in_current_module,          { a "class of" this particular class has been created in            this module          }          classref_created_in_current_module : boolean;          objecttype     : tobjecttyp;          constructor create(ot:tobjecttyp;const n:string;c:tobjectdef;doregister:boolean);virtual;          constructor ppuload(ppufile:tcompilerppufile);          destructor  destroy;override;          function getcopy : tstoreddef;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function GetTypeName:string;override;          procedure buildderef;override;          procedure deref;override;          procedure derefimpl;override;          procedure resetvmtentries;          procedure copyvmtentries(objdef:tobjectdef);          function  getparentdef:tdef;override;          function  size : asizeint;override;          function  alignment:shortint;override;          function  vmtmethodoffset(index:longint):longint;          function  members_need_inittable : boolean;          { this should be called when this class implements an interface }          procedure prepareguid;          function  is_publishable : boolean;override;          function  needs_inittable : boolean;override;          function  needs_separate_initrtti : boolean;override;          function  has_non_trivial_init_child(check_parent:boolean):boolean;override;          function  rtti_mangledname(rt:trttitype):TSymStr;override;          function  vmt_mangledname : TSymStr;          function  vmt_def: trecorddef;          procedure check_forwards; override;          procedure insertvmt;          function  vmt_offset: asizeint;          procedure set_parent(c : tobjectdef);          function find_destructor: tprocdef;          function implements_any_interfaces: boolean;          { dispinterface support }          function get_next_dispid: longint;          { enumerator support }          function search_enumerator_get: tprocdef; override;          function search_enumerator_move: tprocdef; override;          function search_enumerator_current: tsym; override;          { WPO }          procedure register_created_object_type;override;          procedure register_maybe_created_object_type;          procedure register_created_classref_type;          procedure register_vmt_call(index:longint);          { ObjC }          procedure finish_objc_data;          function check_objc_types: boolean;          { C++ }          procedure finish_cpp_data;       end;       tobjectdefclass = class of tobjectdef;       tclassrefdef = class(tabstractpointerdef)          constructor create(def:tdef);virtual;          constructor ppuload(ppufile:tcompilerppufile);          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function getcopy:tstoreddef;override;          function GetTypeName:string;override;          function is_publishable : boolean;override;          function rtti_mangledname(rt:trttitype):TSymStr;override;          procedure register_created_object_type;override;       end;       tclassrefdefclass = class of tclassrefdef;       tarraydef = class(tstoreddef)          lowrange,          highrange     : asizeint;          rangedef      : tdef;          rangedefderef : tderef;          arrayoptions  : tarraydefoptions;          symtable      : TSymtable;       protected          _elementdef      : tdef;          _elementdefderef : tderef;          procedure setelementdef(def:tdef);       public          function elesize : asizeint;          function elepackedbitsize : asizeint;          function elecount : asizeuint;          constructor create_from_pointer(def:tpointerdef);virtual;          constructor create(l,h:asizeint;def:tdef);virtual;          constructor create_openarray;virtual;          class function getreusable(def: tdef; elems: asizeint): tarraydef; virtual;          { same as above, but in case the def must never be freed after the            current module has been compiled -- even if the def was not written            to the ppu file (for defs in para locations, as we don't reset them            so we don't have to recalculate them all the time) }          class function getreusable_no_free(def: tdef; elems: asizeint): tarraydef;          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy; override;          function getcopy : tstoreddef;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  GetTypeName:string;override;          function  getmangledparaname : TSymStr;override;          procedure buildderef;override;          procedure deref;override;          function size : asizeint;override;          function alignment : shortint;override;          { returns the label of the range check string }          function needs_inittable : boolean;override;          function needs_separate_initrtti : boolean;override;          property elementdef : tdef read _elementdef write setelementdef;          function is_publishable : boolean;override;       end;       tarraydefclass = class of tarraydef;       torddef = class(tstoreddef)          low,high : TConstExprInt;          ordtype  : tordtype;          constructor create(t : tordtype;v,b : TConstExprInt; doregister: boolean);virtual;          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  is_publishable : boolean;override;          function  GetTypeName:string;override;          function alignment:shortint;override;          procedure setsize;          function  packedbitsize: asizeint; override;          function getvardef : longint;override;       end;       torddefclass = class of torddef;       tfloatdef = class(tstoreddef)          floattype : tfloattype;          constructor create(t: tfloattype; doregister: boolean);virtual;          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  GetTypeName:string;override;          function  is_publishable : boolean;override;          function alignment:shortint;override;          function structalignment: shortint;override;          procedure setsize;          function  getvardef:longint;override;       end;       tfloatdefclass = class of tfloatdef;       { tabstractprocdef }       tprocnameoption = (pno_showhidden, pno_proctypeoption, pno_paranames,         pno_ownername, pno_noclassmarker, pno_noleadingdollar,         pno_mangledname, pno_noparams);       tprocnameoptions = set of tprocnameoption;       tproccopytyp = (pc_normal,                       { everything except for hidden parameters }                       pc_normal_no_hidden,                       { always creates a top-level function, removes all                         special parameters (self, vmt, parentfp, ...) }                       pc_bareproc,                       { creates a procvardef describing only the code pointer                         of a method/netsted function/... }                       pc_address_only                       );       tabstractprocdef = class(tstoreddef)          { saves a definition to the return type }          returndef       : tdef;          returndefderef  : tderef;          parast          : TSymtable;          paras           : tparalist;          proctypeoption  : tproctypeoption;          proccalloption  : tproccalloption;          procoptions     : tprocoptions;          callerargareasize,          calleeargareasize: pint;{$ifdef m68k}          exp_funcretloc : tregister;   { explicit funcretloc for AmigaOS }{$endif}          funcretloc : array[callerside..calleeside] of TCGPara;          has_paraloc_info : tcallercallee; { paraloc info is available }          { number of user visible parameters }          maxparacount,          minparacount    : byte;          constructor create(dt:tdeftyp;level:byte;doregister:boolean);          constructor ppuload(dt:tdeftyp;ppufile:tcompilerppufile);          destructor destroy;override;          procedure  ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          procedure calcparas;          function mangledprocparanames(oldlen : longint) : string;          function  typename_paras(pno: tprocnameoptions): ansistring;          function  is_methodpointer:boolean;virtual;          function  is_addressonly:boolean;virtual;          function  no_self_node:boolean;          { get either a copy as a procdef or procvardef }          function  getcopyas(newtyp:tdeftyp;copytyp:tproccopytyp; const paraprefix: string): tstoreddef; virtual;          function  compatible_with_pointerdef_size(ptr: tpointerdef): boolean; virtual;          procedure check_mark_as_nested;          procedure init_paraloc_info(side: tcallercallee);          procedure done_paraloc_info(side: tcallercallee);          function stack_tainting_parameter(side: tcallercallee): boolean;          function is_pushleftright: boolean;virtual;          function address_type:tdef;virtual;          { address type, generated for ofs() }          function ofs_address_type:tdef;virtual;          procedure declared_far;virtual;          procedure declared_near;virtual;       private          procedure count_para(p:TObject;arg:pointer);          procedure insert_para(p:TObject;arg:pointer);       end;       tprocvardef = class(tabstractprocdef)          constructor create(level:byte);virtual;          { returns a procvardef that represents the address of a proc(var)def }          class function getreusableprocaddr(def: tabstractprocdef): tprocvardef; virtual;          { same as above, but in case the def must never be freed after the            current module has been compiled -- even if the def was not written            to the ppu file (for defs in para locations, as we don't reset them            so we don't have to recalculate them all the time) }          class function getreusableprocaddr_no_free(def: tabstractprocdef): tprocvardef;          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  GetSymtable(t:tGetSymtable):TSymtable;override;          function  size : asizeint;override;          function  GetTypeName:string;override;          function  is_publishable : boolean;override;          function  is_methodpointer:boolean;override;          function  is_addressonly:boolean;override;          function  getmangledparaname:TSymStr;override;          function getcopyas(newtyp: tdeftyp; copytyp: tproccopytyp; const paraprefix: string): tstoreddef; override;       end;       tprocvardefclass = class of tprocvardef;       tmessageinf = record         case integer of           0 : (str : pshortstring);           1 : (i : longint);       end;       tinlininginfo = record          { node tree }          code  : tnode;          flags : tprocinfoflags;       end;       pinlininginfo = ^tinlininginfo;{$ifdef oldregvars}       { register variables }       pregvarinfo = ^tregvarinfo;       tregvarinfo = record          regvars : array[1..maxvarregs] of tsym;          regvars_para : array[1..maxvarregs] of boolean;          regvars_refs : array[1..maxvarregs] of longint;          fpuregvars : array[1..maxfpuvarregs] of tsym;          fpuregvars_para : array[1..maxfpuvarregs] of boolean;          fpuregvars_refs : array[1..maxfpuvarregs] of longint;       end;{$endif oldregvars}       timplprocdefinfo = record          resultname : pshortstring;          parentfpstruct: tsym;          parentfpstructptrtype: tdef;          parentfpinitblock: tnode;          procstarttai,          procendtai   : tai;          skpara: pointer;          personality: tprocdef;          forwarddef,          interfacedef : boolean;          hasforward  : boolean;       end;       pimplprocdefinfo = ^timplprocdefinfo;       { tprocdef }       tprocdef = class(tabstractprocdef)       protected{$ifdef symansistr}         _mangledname : ansistring;{$else symansistr}         _mangledname : pshortstring;{$endif}         { information that is only required until the implementation of the           procdef has been handled }         implprocdefinfo : pimplprocdefinfo;         function GetResultName: PShortString;         procedure SetResultName(AValue: PShortString);         function GetParentFPStruct: tsym;         procedure SetParentFPStruct(AValue: tsym);         function GetParentFPStructPtrType: tdef;         procedure SetParentFPStructPtrType(AValue: tdef);         function GetParentFPInitBlock: tnode;         procedure SetParentFPInitBlock(AValue: tnode);         function Getprocstarttai: tai;         procedure Setprocstarttai(AValue: tai);         function Getprocendtai: tai;         procedure Setprocendtai(AValue: tai);         function Getskpara: pointer;         procedure Setskpara(AValue: pointer);         function Getpersonality: tprocdef;         procedure Setpersonality(AValue: tprocdef);         function Getforwarddef: boolean;         procedure Setforwarddef(AValue: boolean);         function Getinterfacedef: boolean;         procedure Setinterfacedef(AValue: boolean);virtual;         function Gethasforward: boolean;         procedure Sethasforward(AValue: boolean);         function GetIsEmpty: boolean;         procedure SetIsEmpty(AValue: boolean);         function GetHasInliningInfo: boolean;         procedure SetHasInliningInfo(AValue: boolean);       public          messageinf : tmessageinf;          dispid : longint;{$ifndef EXTDEBUG}          { where is this function defined and what were the symbol            flags, needed here because there            is only one symbol for all overloaded functions            EXTDEBUG has fileinfo in tdef (PFV) }          fileinfo : tfileposinfo;{$endif}          symoptions : tsymoptions;          deprecatedmsg : pshortstring;          { generic support }          genericdecltokenbuf : tdynamicarray;          { symbol owning this definition }          procsym : tsym;          procsymderef : tderef;          { alias names }          aliasnames : TCmdStrList;          { symtables }          localst : TSymtable;          funcretsym : tsym;          funcretsymderef : tderef;          struct : tabstractrecorddef;          structderef : tderef;          implprocoptions: timplprocoptions;          { import info }          import_dll,          import_name : pshortstring;          { info for inlining the subroutine, if this pointer is nil,            the procedure can't be inlined }          inlininginfo : pinlininginfo;{$ifdef oldregvars}          regvarinfo: pregvarinfo;{$endif oldregvars}          import_nr    : word;          extnumber    : word;          { set to a value different from tsk_none in case this procdef is for            a routine that has to be internally generated by the compiler }          synthetickind: tsynthetickind;          visibility   : tvisibility;          constructor create(level:byte;doregister:boolean);virtual;          constructor ppuload(ppufile:tcompilerppufile);          destructor  destroy;override;          procedure   freeimplprocdefinfo;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          procedure buildderef;override;          procedure buildderefimpl;override;          procedure deref;override;          procedure derefimpl;override;          function  GetSymtable(t:tGetSymtable):TSymtable;override;          { warnings:              * the symtablestack top has to be the symtable to which the copy                should be added              * getcopy does not create a finished/ready-to-use procdef; it                needs to be finalised afterwards by calling                symcreat.finish_copied_procdef() afterwards          }          function  getcopyas(newtyp:tdeftyp;copytyp:tproccopytyp; const paraprefix: string): tstoreddef; override;          function  getcopy: tstoreddef; override;          function  GetTypeName : string;override;          function  mangledname : TSymStr; virtual;          procedure setmangledname(const s : TSymStr);          procedure setcompilerprocname;          function  fullprocname(showhidden:boolean):string;          function  customprocname(pno: tprocnameoptions):ansistring;          function  defaultmangledname: TSymStr;          function  cplusplusmangledname : TSymStr;          function  objcmangledname : TSymStr;          function  is_methodpointer:boolean;override;          function  is_addressonly:boolean;override;          procedure make_external;          procedure init_genericdecl;          function getfuncretsyminfo(out ressym: tsym; out resdef: tdef): boolean; virtual;          { returns whether the mangled name or any of its aliases is equal to            s }          function  has_alias_name(const s: TSymStr):boolean;          { aliases to fields only required when a function is implemented in            the current unit }          property resultname: PShortString read GetResultName write SetResultName;          { temporary reference to structure containing copies of all local            variables and parameters accessed by nested routines; reference to            this structure is passed as "parent frame pointer" on targets that            lack this concept (at least JVM and LLVM); no need to save to/            restore from ppu, since nested routines are always in the same            unit }          property parentfpstruct: tsym read GetParentFPStruct write SetParentFPStruct;          { pointer to parentfpstruct's type (not yet valid during parsing, so            cannot be used for $parentfp parameter) (no need to save to ppu) }          property parentfpstructptrtype: tdef read GetParentFPStructPtrType write SetParentFPStructPtrType;          { code to copy the parameters accessed from nested routines into the            parentfpstruct (no need to save to ppu) }          property parentfpinitblock: tnode read GetParentFPInitBlock write SetParentFPInitBlock;          { First/last assembler symbol/instruction in aasmoutput list.            Note: initialised after compiling the code for the procdef, but              not saved to/restored from ppu. Used when inserting debug info }          property procstarttai: tai read Getprocstarttai write Setprocstarttai;          property procendtai: tai read Getprocendtai write Setprocendtai;          { optional parameter for the synthetic routine generation logic }          property skpara: pointer read Getskpara write Setskpara;          { ABI-conformant exception handling personality function }          property personality: tprocdef read Getpersonality write Setpersonality;          { true, if the procedure is only declared            (forward procedure) }          property forwarddef: boolean read Getforwarddef write Setforwarddef;          { true if the procedure is declared in the interface }          property interfacedef: boolean read Getinterfacedef write Setinterfacedef;          { true if the procedure has a forward declaration }          property hasforward: boolean read Gethasforward write Sethasforward;          { true if the routine's body is empty }          property isempty: boolean read GetIsEmpty write SetIsEmpty;          { true if all information required to inline this routine is available }          property has_inlininginfo: boolean read GetHasInliningInfo write SetHasInliningInfo;       end;       tprocdefclass = class of tprocdef;       { single linked list of overloaded procs }       pprocdeflist = ^tprocdeflist;       tprocdeflist = record         def  : tprocdef;         defderef : tderef;         next : pprocdeflist;       end;       tstringdef = class(tstoreddef)          encoding   : tstringencoding;          stringtype : tstringtype;          len        : asizeint;          constructor createshort(l: byte; doregister: boolean);virtual;          constructor loadshort(ppufile:tcompilerppufile);          constructor createlong(l: asizeint; doregister: boolean);virtual;          constructor loadlong(ppufile:tcompilerppufile);          constructor createansi(aencoding: tstringencoding; doregister: boolean);virtual;          constructor loadansi(ppufile:tcompilerppufile);          constructor createwide(doregister: boolean);virtual;          constructor loadwide(ppufile:tcompilerppufile);          constructor createunicode(doregister: boolean);virtual;          constructor loadunicode(ppufile:tcompilerppufile);virtual;          function getcopy : tstoreddef;override;          function  stringtypname:string;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          function  GetTypeName:string;override;          function  getmangledparaname:TSymStr;override;          function  is_publishable : boolean;override;          function  size:asizeint;override;          function alignment : shortint;override;          function  needs_inittable : boolean;override;          function  getvardef:longint;override;       end;       tstringdefclass = class of tstringdef;       { tenumdef }       tenumdef = class(tstoreddef)          minval,          maxval    : asizeint;          basedef   : tenumdef;          basedefderef : tderef;          symtable  : TSymtable;          has_jumps : boolean;          constructor create;virtual;          constructor create_subrange(_basedef:tenumdef;_min,_max:asizeint);virtual;          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy;override;          function getcopy : tstoreddef;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          procedure buildderef;override;          procedure deref;override;          function  GetTypeName:string;override;          function  is_publishable : boolean;override;          procedure calcsavesize(packenum: shortint);          function  packedbitsize: asizeint; override;          procedure setmax(_max:asizeint);          procedure setmin(_min:asizeint);          function  min:asizeint;          function  max:asizeint;          function  getfirstsym:tsym;          function  int2enumsym(l: asizeint): tsym;          { returns basedef if assigned, otherwise self }          function getbasedef: tenumdef;       end;       tenumdefclass = class of tenumdef;       tsetdef = class(tstoreddef)          elementdef : tdef;          elementdefderef : tderef;          setbase,          setmax   : asizeint;          constructor create(def: tdef; low, high: asizeint; doregister: boolean);virtual;          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          { do not override this routine in platform-specific subclasses,            override ppuwrite_platform instead }          procedure ppuwrite(ppufile:tcompilerppufile);override;final;          procedure buildderef;override;          procedure deref;override;          function  GetTypeName:string;override;          function  is_publishable : boolean;override;          function alignment: shortint; override;       end;       tsetdefclass = class of tsetdef;       tgenericdummyentry = class         dummysym : tsym;         resolvedsym : tsym;       end;       tdefawaresymtablestack = class(TSymtablestack)       private         procedure add_helpers_and_generics(st:tsymtable;addgenerics:boolean);         procedure remove_helpers_and_generics(st:tsymtable);inline;         procedure remove_helpers(st:tsymtable);         procedure remove_generics(st:tsymtable);         procedure pushcommon(st:tsymtable);inline;       public         procedure push(st: TSymtable); override;         procedure pushafter(st,afterst:TSymtable); override;         procedure pop(st: TSymtable); override;       end;    var       current_structdef: tabstractrecorddef; { used for private functions check !! }       current_genericdef: tstoreddef;        { used to reject declaration of generic class inside generic class }       current_specializedef: tstoreddef;     { used to implement usage of generic class in itself }       cfiledef: tfiledefclass;       cvariantdef: tvariantdefclass;       cformaldef: tformaldefclass;       cforwarddef: tforwarddefclass;       cundefineddef: tundefineddefclass;       cerrordef: terrordefclass;       cpointerdef: tpointerdefclass;       crecorddef: trecorddefclass;       cimplementedinterface: timplementedinterfaceclass;       cobjectdef: tobjectdefclass;       cclassrefdef: tclassrefdefclass;       carraydef: tarraydefclass;       corddef: torddefclass;       cfloatdef: tfloatdefclass;       cprocvardef: tprocvardefclass;       cprocdef: tprocdefclass;       cstringdef: tstringdefclass;       cenumdef: tenumdefclass;       csetdef: tsetdefclass;    { default types }       generrordef,               { error in definition }       voidpointertype,           { pointer for Void-pointeddef }       charpointertype,           { pointer for Char-pointeddef }       widecharpointertype,       { pointer for WideChar-pointeddef }       voidcodepointertype,       { pointer to code; corresponds to System.CodePointer }       voidstackpointertype,      { the pointer type used for accessing parameters and local vars on the stack }       parentfpvoidpointertype,   { void pointer with the size of the hidden parentfp parameter, passed to nested functions }{$ifdef x86}       voidnearpointertype,       voidnearcspointertype,       voidneardspointertype,       voidnearsspointertype,       voidnearespointertype,       voidnearfspointertype,       voidneargspointertype,  {$ifdef i8086}       voidfarpointertype,       voidhugepointertype,       charnearpointertype,       charfarpointertype,       charhugepointertype,       bytefarpointertype,        { used for Mem[] }       wordfarpointertype,        { used for MemW[] }       longintfarpointertype,     { used for MemL[] }  {$endif i8086}{$endif x86}       cundefinedtype,       cformaltype,               { unique formal definition }       ctypedformaltype,          { unique typed formal definition }       voidtype,                  { Void (procedure) }       cansichartype,             { Char }       cwidechartype,             { WideChar }       pasbool1type,              { boolean type }       pasbool8type,       pasbool16type,       pasbool32type,       pasbool64type,       bool8type,       bool16type,       bool32type,       bool64type,                { implement me }{$ifdef llvm}       llvmbool1type,             { LLVM i1 type }{$endif llvm}       u8inttype,                 { 8-Bit unsigned integer }       s8inttype,                 { 8-Bit signed integer }       u16inttype,                { 16-Bit unsigned integer }       s16inttype,                { 16-Bit signed integer }       u24inttype,                { 24-Bit unsigned integer }       s24inttype,                { 24-Bit signed integer }       u32inttype,                { 32-Bit unsigned integer }       s32inttype,                { 32-Bit signed integer }       u40inttype,                { 40-Bit unsigned integer }       s40inttype,                { 40-Bit signed integer }       u48inttype,                { 48-Bit unsigned integer }       s48inttype,                { 48-Bit signed integer }       u56inttype,                { 56-Bit unsigned integer }       s56inttype,                { 56-Bit signed integer }       u64inttype,                { 64-bit unsigned integer }       s64inttype,                { 64-bit signed integer }       u128inttype,               { 128-bit unsigned integer }       s128inttype,               { 128-bit signed integer }       s32floattype,              { 32 bit floating point number }       s64floattype,              { 64 bit floating point number }       s80floattype,              { 80 bit floating point number }       sc80floattype,             { 80 bit floating point number but stored like in C }       s64currencytype,           { pointer to a currency type }       cshortstringtype,          { pointer to type of short string const   }       clongstringtype,           { pointer to type of long string const   }       cansistringtype,           { pointer to type of ansi string const  }       cwidestringtype,           { pointer to type of wide string const  }       cunicodestringtype,       openshortstringtype,       { pointer to type of an open shortstring,                                    needed for readln() }       openchararraytype,         { pointer to type of an open array of char,                                    needed for readln() }       cfiletype,                 { get the same definition for all file }                                  { used for stabs }       methodpointertype,         { typecasting of methodpointers to extract self }       nestedprocpointertype,     { typecasting of nestedprocpointers to extract parentfp }       hresultdef,       typekindtype,              { def of TTypeKind for correct handling of GetTypeKind parameters }       { we use only one variant def for every variant class }       cvarianttype,       colevarianttype,       { default integer type, normally s32inttype on 32 bit systems and s64bittype on 64 bit systems }       sinttype,       uinttype,       { integer types corresponding to OS_SINT/OS_INT }       ossinttype,       osuinttype,       { integer types corresponding to the ALU size, sizeof(aint) and the ALUSInt/ALUUInt types in the system unit }       alusinttype,       aluuinttype,       { integer types corresponding to SizeInt and SizeUInt for the target platform }       sizeuinttype,       sizesinttype,       { unsigned and signed ord type with the same size as a pointer }       ptruinttype,       ptrsinttype,       { unsigned and signed ord type with the same size as a codepointer }       codeptruinttype,       codeptrsinttype,       { several types to simulate more or less C++ objects for GDB }       vmttype,       vmtarraytype,       pvmttype      : tdef;     { type of classrefs, used for stabs }       { pointer to the anchestor of all classes }       class_tobject : tobjectdef;       { pointer to the ancestor of all COM interfaces }       interface_iunknown : tobjectdef;       { pointer to the ancestor of all dispinterfaces }       interface_idispatch : tobjectdef;       { pointer to the TGUID type         of all interfaces         }       rec_tguid : trecorddef;       { jump buffer type, used by setjmp }       rec_jmp_buf : trecorddef;       { system.texceptaddr type, used by fpc_pushexceptaddr }       rec_exceptaddr: trecorddef;       { Objective-C base types }       objc_metaclasstype,       objc_superclasstype,       objc_idtype,       objc_seltype              : tpointerdef;       objc_objecttype           : trecorddef;       { base type of @protocol(protocolname) Objective-C statements }       objc_protocoltype         : tobjectdef;       { helper types for for-in "fast enumeration" support in Objective-C 2.0 }       objc_fastenumeration      : tobjectdef;       objc_fastenumerationstate : trecorddef;{$ifdef llvm}       { llvm types }       { a unique def to identify any kind of metadata }       llvm_metadatatype         : tdef;{$endif llvm}       { Java base types }       { java.lang.Object }       java_jlobject             : tobjectdef;       { java.lang.Throwable }       java_jlthrowable          : tobjectdef;       { FPC base type for records }       java_fpcbaserecordtype    : tobjectdef;       { java.lang.String }       java_jlstring             : tobjectdef;       { java.lang.Enum }       java_jlenum               : tobjectdef;       { java.util.EnumSet }       java_juenumset            : tobjectdef;       { java.util.BitSet }       java_jubitset             : tobjectdef;       { FPC java implementation of ansistrings }       java_ansistring           : tobjectdef;       { FPC java implementation of shortstrings }       java_shortstring          : tobjectdef;       { FPC java procvar base class }       java_procvarbase          : tobjectdef;    function make_mangledname(const typeprefix:TSymStr;st:TSymtable;const suffix:TSymStr):TSymStr;    function make_dllmangledname(const dllname,importname:TSymStr;                                 import_nr : word; pco : tproccalloption):TSymStr;    { should be in the types unit, but the types unit uses the node stuff :( }    function is_interfacecom(def: tdef): boolean;    function is_interfacecom_or_dispinterface(def: tdef): boolean;    function is_any_interface_kind(def: tdef): boolean;    function is_interfacecorba(def: tdef): boolean;    function is_interface(def: tdef): boolean;    function is_dispinterface(def: tdef): boolean;    function is_object(def: tdef): boolean;    function is_class(def: tdef): boolean;    function is_cppclass(def: tdef): boolean;    function is_objectpascal_helper(def: tdef): boolean;    function is_objcclass(def: tdef): boolean;    function is_objcclassref(def: tdef): boolean;    function is_objcprotocol(def: tdef): boolean;    function is_objccategory(def: tdef): boolean;    function is_objc_class_or_protocol(def: tdef): boolean;    function is_objc_protocol_or_category(def: tdef): boolean;    function is_classhelper(def: tdef): boolean;    function is_class_or_interface(def: tdef): boolean;    function is_class_or_interface_or_objc(def: tdef): boolean;    function is_class_or_interface_or_objc_or_java(def: tdef): boolean;    function is_class_or_interface_or_dispinterface_or_objc_or_java(def: tdef): boolean;    function is_class_or_interface_or_object(def: tdef): boolean;    function is_class_or_interface_or_dispinterface(def: tdef): boolean;    function is_implicit_pointer_object_type(def: tdef): boolean;    { returns true, if def is a type which is an implicit pointer to an array (dyn. array or dyn. string) }    function is_implicit_array_pointer(def: tdef): boolean;    function is_class_or_object(def: tdef): boolean;    function is_record(def: tdef): boolean;    function is_javaclass(def: tdef): boolean;    function is_javaclassref(def: tdef): boolean;    function is_javainterface(def: tdef): boolean;    function is_java_class_or_interface(def: tdef): boolean;    procedure loadobjctypes;    procedure maybeloadcocoatypes;    function use_vectorfpu(def : tdef) : boolean;    function getansistringcodepage:tstringencoding; inline;    function getansistringdef:tstringdef;    function getparaencoding(def:tdef):tstringencoding; inline;    function get_threadvar_record(def: tdef; out index_field, non_mt_data_field: tsym): trecorddef;    function get_recorddef(prefix:tinternaltypeprefix;const fields:array of tdef; packrecords:shortint): trecorddef;    { get a table def of the form        record          count: countdef;          elements: array[0..count-1] of elementdef        end;      Returns both the outer record and the inner arraydef    }    procedure get_tabledef(prefix:tinternaltypeprefix;countdef,elementdef:tdef;count:longint;packrecords:shortint;out recdef:trecorddef;out arrdef:tarraydef);implementation    uses      SysUtils,      cutils,      { global }      verbose,      { target }      systems,paramgr,      { symtable }      symsym,symtable,defutil,objcdef,      { parser }      pgenutil,      { module }      fmodule,      { other }      gendef,      fpccrc,      entfile      ;{****************************************************************************                                  Helpers****************************************************************************}    function getansistringcodepage:tstringencoding; inline;      begin        if ([cs_explicit_codepage,cs_system_codepage]*current_settings.moduleswitches)<>[] then          result:=current_settings.sourcecodepage        else          result:=0;      end;    function getansistringdef:tstringdef;      var        symtable:tsymtable;        oldstack : tsymtablestack;      begin        { if a codepage is explicitly defined in this mudule we need to return          a replacement for ansistring def }        if ([cs_explicit_codepage,cs_system_codepage]*current_settings.moduleswitches)<>[] then          begin            if not assigned(current_module) then              internalerror(2011101301);            { codepage can be redeclared only once per unit so we don't need a list of              redefined ansistring but only one pointer }            if not assigned(current_module.ansistrdef) then              begin                { if we did not create it yet we need to do this now }                if current_module.in_interface then                  symtable:=current_module.globalsymtable                else                  symtable:=current_module.localsymtable;                { create a temporary stack as it's not good (TM) to mess around                  with the order if the unit contains generics or helpers; don't                  use a def aware symtablestack though }                oldstack:=symtablestack;                symtablestack:=tsymtablestack.create;                symtablestack.push(symtable);                current_module.ansistrdef:=cstringdef.createansi(current_settings.sourcecodepage,true);                symtablestack.pop(symtable);                symtablestack.free;                symtablestack:=oldstack;              end;            result:=tstringdef(current_module.ansistrdef);          end        else          result:=tstringdef(cansistringtype);      end;    function getparaencoding(def:tdef):tstringencoding; inline;      begin        { don't pass CP_NONE encoding to internal functions          they expect 0 encoding instead          exception: result of string concatenation, because if you pass the          result of a string concatenation to a rawbytestring, the result of          that concatenation shouldn't be converted to defaultsystemcodepage          if all strings have the same type }        result:=tstringdef(def).encoding;        if result=globals.CP_NONE then          result:=0      end;    function get_threadvar_record(def: tdef; out index_field, non_mt_data_field: tsym): trecorddef;      var        typ: ttypesym;        name: string;      begin        name:=internaltypeprefixName[itp_threadvar_record]+def.unique_id_str;        typ:=try_search_current_module_type(name);        if assigned(typ) then          begin            result:=trecorddef(ttypesym(typ).typedef);            index_field:=tsym(result.symtable.symlist[0]);            non_mt_data_field:=tsym(result.symtable.symlist[1]);            exit;          end;        { set recordalinmin to sizeof(pint), so the second field gets put at          offset = sizeof(pint) as expected }        result:=crecorddef.create_global_internal(          name,sizeof(pint),sizeof(pint),          init_settings.alignment.maxCrecordalign);{$ifdef cpu16bitaddr}        index_field:=result.add_field_by_def('',u16inttype);{$else cpu16bitaddr}        index_field:=result.add_field_by_def('',u32inttype);{$endif cpu16bitaddr}        non_mt_data_field:=result.add_field_by_def('',def);        { no need to add alignment padding, we won't create arrays of these }      end;    function get_recorddef(prefix:tinternaltypeprefix; const fields:array of tdef; packrecords:shortint): trecorddef;      var        fieldlist: tfplist;        srsym: tsym;        srsymtable: tsymtable;        i: longint;        name : TIDString;      begin        name:=copy(internaltypeprefixName[prefix],2,length(internaltypeprefixName[prefix]));        if searchsym_type(name,srsym,srsymtable) then          begin            result:=trecorddef(ttypesym(srsym).typedef);            exit          end;        { also always search in the current module (symtables are popped for          RTTI related code already) }        if searchsym_in_module(pointer(current_module),name,srsym,srsymtable) then          begin            result:=trecorddef(ttypesym(srsym).typedef);            exit;          end;        fieldlist:=tfplist.create;        for i:=low(fields) to high(fields) do          fieldlist.add(fields[i]);        result:=crecorddef.create_global_internal(internaltypeprefixName[prefix],packrecords,          targetinfos[target_info.system]^.alignment.recordalignmin,          targetinfos[target_info.system]^.alignment.maxCrecordalign);        result.add_fields_from_deflist(fieldlist);        fieldlist.free;      end;    procedure get_tabledef(prefix:tinternaltypeprefix;countdef,elementdef:tdef;count:longint;packrecords:shortint;out recdef:trecorddef;out arrdef:tarraydef);      var        fields: tfplist;        name: TIDString;        srsym: tsym;        srsymtable: tsymtable;      begin        { already created a message string table with this number of elements          in this unit -> reuse the def }        name:=internaltypeprefixName[prefix]+tostr(count);        if searchsym_type(copy(name,2,length(name)),srsym,srsymtable) then          begin            recdef:=trecorddef(ttypesym(srsym).typedef);            arrdef:=tarraydef(trecordsymtable(recdef.symtable).findfieldbyoffset(countdef.size).vardef);            exit          end;        { also always search in the current module (symtables are popped for          RTTI related code already) }        if searchsym_in_module(pointer(current_module),copy(name,2,length(name)),srsym,srsymtable) then          begin            recdef:=trecorddef(ttypesym(srsym).typedef);            arrdef:=tarraydef(trecordsymtable(recdef.symtable).findfieldbyoffset(countdef.size).vardef);            exit;          end;        recdef:=crecorddef.create_global_internal(name,packrecords,          targetinfos[target_info.system]^.alignment.recordalignmin,          targetinfos[target_info.system]^.alignment.maxCrecordalign);        fields:=tfplist.create;        fields.add(countdef);        if count>0 then          begin            arrdef:=carraydef.create(0,count-1,sizeuinttype);            arrdef.elementdef:=elementdef;            fields.add(arrdef);          end        else          arrdef:=nil;        recdef.add_fields_from_deflist(fields);        fields.free;      end;    function make_mangledname(const typeprefix:TSymStr;st:TSymtable;const suffix:TSymStr):TSymStr;      var        s,        prefix : TSymStr;        crc : dword;      begin        prefix:='';        if not assigned(st) then         internalerror(200204212);        { sub procedures }        while (st.symtabletype in [localsymtable,parasymtable]) do         begin           if st.defowner.typ<>procdef then            internalerror(200204173);           { Add the full mangledname of procedure to prevent             conflicts with 2 overloads having both a nested procedure             with the same name, see tb0314 (PFV) }           s:=tprocdef(st.defowner).procsym.name;           s:=s+tprocdef(st.defowner).mangledprocparanames(Length(s));           if prefix<>'' then             prefix:=s+'_'+prefix           else             prefix:=s;           if length(prefix)>100 then             begin               crc:=0;               crc:=UpdateCrc32(crc,prefix[1],length(prefix));               prefix:='$CRC'+hexstr(crc,8);             end;           st:=st.defowner.owner;         end;        { object/classes symtable, nested type definitions in classes require the while loop }        while st.symtabletype in [ObjectSymtable,recordsymtable] do         begin           if not (st.defowner.typ in [objectdef,recorddef]) then            internalerror(200204174);           prefix:=tabstractrecorddef(st.defowner).objname^+'_$_'+prefix;           st:=st.defowner.owner;         end;        { symtable must now be static or global }        if not(st.symtabletype in [staticsymtable,globalsymtable]) then          internalerror(200204175);        { The mangled name is made out of at most 4 parts:         1) Optional typeprefix given as first parameter            with '_$' appended if not empty         2) Unit name or 'P$'+program name (never empty)         3) optional prefix variable that contains a unique            name for the local symbol table (prepended with '$_$'            if not empty)         4) suffix as given as third parameter,            also optional (i.e. can be empty)            prepended by '_$$_' if not empty }        result:='';        if typeprefix<>'' then          result:=result+typeprefix+'_$';        { Add P$ for program, which can have the same name as          a unit }        if (TSymtable(main_module.localsymtable)=st) and           (not main_module.is_unit) then          result:=result+'P$'+st.name^        else          result:=result+st.name^;        if prefix<>'' then          result:=result+'$_$'+prefix;        if suffix<>'' then          result:=result+'_$$_'+suffix;        { the Darwin assembler assumes that all symbols starting with 'L' are local }        { Further, the Mac OS X 10.5 linker does not consider symbols which do not  }        { start with '_' as regular symbols (it does not generate N_GSYM entries    }        { those in the debug map, leading to troubles with dsymutil). So always     }        { add an underscore on darwin.                                              }        if (target_info.system in systems_darwin) then          result := '_' + result;      end;    function make_dllmangledname(const dllname,importname:TSymStr;import_nr : word; pco : tproccalloption):TSymStr;       var         crc : cardinal;         i : longint;         use_crc : boolean;         dllprefix : TSymStr;      begin        if (target_info.system in (systems_all_windows + systems_nativent +                           [system_i386_emx, system_i386_os2]))            and (dllname <> '') then          begin            dllprefix:=lower(ExtractFileName(dllname));            { Remove .dll suffix if present }            if copy(dllprefix,length(dllprefix)-3,length(dllprefix))='.dll' then              dllprefix:=copy(dllprefix,1,length(dllprefix)-4);            use_crc:=false;            for i:=1 to length(dllprefix) do              if not (dllprefix[i] in ['a'..'z','A'..'Z','_','0'..'9']) then                begin                  use_crc:=true;                  break;                end;            if use_crc then              begin                crc:=0;                crc:=UpdateCrc32(crc,dllprefix[1],length(dllprefix));                dllprefix:='_$dll$crc$'+hexstr(crc,8)+'$';              end            else              dllprefix:='_$dll$'+dllprefix+'$';            if importname<>'' then              result:=dllprefix+importname            else              result:=dllprefix+'_index_'+tostr(import_nr);            { Replace ? and @ in import name, since GNU AS does not allow these characters in symbol names. }            { This allows to import VC++ mangled names from DLLs. }            { Do not perform replacement, if external symbol is not imported from DLL. }            if (dllname<>'') then              begin                Replace(result,'?','__q$$');    {$ifdef arm}                { @ symbol is not allowed in ARM assembler only }                Replace(result,'@','__a$$');    {$endif arm}             end;          end        else          begin            if importname<>'' then             begin               if not(pco in [pocall_cdecl,pocall_cppdecl]) then                 result:=importname               else                 result:=target_info.Cprefix+importname;             end            else              result:='_index_'+tostr(import_nr);          end;      end;{****************************************************************************           TDEFAWARESYMTABLESTACK           (symtablestack descendant that does some special actions on           the pushed/popped symtables)****************************************************************************}    procedure tdefawaresymtablestack.add_helpers_and_generics(st:tsymtable;addgenerics:boolean);      var        i: integer;        s: string;        list: TFPObjectList;        def: tdef;        sym : tsym;      begin        { search the symtable from first to last; the helper to use will be the          last one in the list }        for i:=0 to st.symlist.count-1 do          begin            sym:=tsym(st.symlist[i]);            if not (sym.typ in [typesym,procsym]) then              continue;            if sym.typ=typesym then              def:=ttypesym(st.SymList[i]).typedef            else              def:=nil;            if is_objectpascal_helper(def) then              begin                s:=generate_objectpascal_helper_key(tobjectdef(def).extendeddef);                Message1(sym_d_adding_helper_for,s);                list:=TFPObjectList(current_module.extendeddefs.Find(s));                if not assigned(list) then                  begin                    list:=TFPObjectList.Create(false);                    current_module.extendeddefs.Add(s,list);                  end;                list.Add(def);              end            else              begin                if addgenerics then                  add_generic_dummysym(sym);                { add nested helpers as well }                if assigned(def) and                    (def.typ in [recorddef,objectdef]) and                    (sto_has_helper in tabstractrecorddef(def).symtable.tableoptions) then                  add_helpers_and_generics(tabstractrecorddef(def).symtable,false);              end;          end;      end;    procedure tdefawaresymtablestack.remove_helpers_and_generics(st:tsymtable);      begin        if sto_has_helper in st.tableoptions then          remove_helpers(st);        if sto_has_generic in st.tableoptions then          remove_generics(st);      end;    procedure tdefawaresymtablestack.remove_helpers(st:TSymtable);      var        i, j: integer;        tmpst: TSymtable;        list: TFPObjectList;      begin        for i:=current_module.extendeddefs.count-1 downto 0 do          begin            list:=TFPObjectList(current_module.extendeddefs[i]);            for j:=list.count-1 downto 0 do              begin                if not (list[j] is tobjectdef) then                  Internalerror(2011031501);                tmpst:=tobjectdef(list[j]).owner;                repeat                  if tmpst=st then                    begin                      list.delete(j);                      break;                    end                  else                    begin                      if assigned(tmpst.defowner) then                        tmpst:=tmpst.defowner.owner                      else                        tmpst:=nil;                    end;                until not assigned(tmpst) or (tmpst.symtabletype in [globalsymtable,staticsymtable]);              end;            if list.count=0 then              current_module.extendeddefs.delete(i);          end;      end;    procedure tdefawaresymtablestack.remove_generics(st:tsymtable);      var        i,j : longint;        entry : tgenericdummyentry;        list : tfpobjectlist;      begin        for i:=current_module.genericdummysyms.count-1 downto 0 do          begin            list:=tfpobjectlist(current_module.genericdummysyms[i]);            if not assigned(list) then              continue;            for j:=list.count-1 downto 0 do              begin                entry:=tgenericdummyentry(list[j]);                if entry.dummysym.owner=st then                  list.delete(j);              end;            if list.count=0 then              current_module.genericdummysyms.delete(i);          end;      end;    procedure tdefawaresymtablestack.pushcommon(st:tsymtable);      begin        if (sto_has_generic in st.tableoptions) or            (              (st.symtabletype in [globalsymtable,staticsymtable]) and              (sto_has_helper in st.tableoptions)            ) then          { nested helpers will be added as well }          add_helpers_and_generics(st,true);      end;    procedure tdefawaresymtablestack.push(st: TSymtable);      begin        pushcommon(st);        inherited push(st);      end;    procedure tdefawaresymtablestack.pushafter(st,afterst:TSymtable);      begin        pushcommon(st);        inherited pushafter(st,afterst);      end;    procedure tdefawaresymtablestack.pop(st: TSymtable);      begin        inherited pop(st);        if (sto_has_generic in st.tableoptions) or            (              (st.symtabletype in [globalsymtable,staticsymtable]) and              (sto_has_helper in st.tableoptions)            ) then          { nested helpers will be removed as well }          remove_helpers_and_generics(st);      end;{****************************************************************************                     TDEF (base class for definitions)****************************************************************************}    constructor tgenericconstraintdata.create;      begin        interfaces:=tfpobjectlist.create(false);        interfacesderef:=tfplist.create;      end;    destructor tgenericconstraintdata.destroy;      var        i : longint;      begin        for i:=0 to interfacesderef.count-1 do          dispose(pderef(interfacesderef[i]));        interfacesderef.free;        interfaces.free;        inherited destroy;      end;    procedure tgenericconstraintdata.ppuload(ppufile: tcompilerppufile);      var        cnt,i : longint;        intfderef : pderef;      begin        ppufile.getsmallset(flags);        cnt:=ppufile.getlongint;        for i:=0 to cnt-1 do          begin            new(intfderef);            ppufile.getderef(intfderef^);            interfacesderef.add(intfderef);          end;      end;    procedure tgenericconstraintdata.ppuwrite(ppufile: tcompilerppufile);      var        i : longint;      begin        ppufile.putsmallset(flags);        ppufile.putlongint(interfacesderef.count);        for i:=0 to interfacesderef.count-1 do          ppufile.putderef(pderef(interfacesderef[i])^);      end;    procedure tgenericconstraintdata.buildderef;      var        intfderef : pderef;        i : longint;      begin        for i:=0 to interfaces.count-1 do          begin            new(intfderef);            intfderef^.build(tobjectdef(interfaces[i]));            interfacesderef.add(intfderef);          end;      end;    procedure tgenericconstraintdata.deref;      var        i : longint;      begin        for i:=0 to interfacesderef.count-1 do          interfaces.add(pderef(interfacesderef[i])^.resolve);      end;    procedure tstoreddef.writeentry(ppufile: tcompilerppufile; ibnr: byte);      begin        ppuwrite_platform(ppufile);        ppufile.writeentry(ibnr);      end;    procedure tstoreddef.ppuwrite_platform(ppufile: tcompilerppufile);      begin        { by default: do nothing }      end;    procedure tstoreddef.ppuload_platform(ppufile: tcompilerppufile);      begin        { by default: do nothing }      end;    class procedure tstoreddef.setup_reusable_def(origdef, newdef: tdef; res: PHashSetItem; oldsymtablestack: tsymtablestack);      var        reusablesymtab: tsymtable;      begin        { must not yet belong to a symtable }        if assigned(newdef.owner) then          internalerror(2015111503);        reusablesymtab:=origdef.getreusablesymtab;        res^.Data:=newdef;        reusablesymtab.insertdef(newdef);        symtablestack:=oldsymtablestack;      end;    constructor tstoreddef.create(dt:tdeftyp;doregister:boolean);      begin         inherited create(dt);         savesize := 0;{$ifdef EXTDEBUG}         fileinfo := current_filepos;{$endif}         generictokenbuf:=nil;         genericdef:=nil;         typesymderef.reset;         genericdefderef.reset;         { Don't register forwarddefs, they are disposed at the           end of an type block }         if (dt=forwarddef) then           exit;         { Register in symtable stack }         if doregister then           begin             { immediately register interface defs, as they will always be               written to the ppu, their defid inlfuences the interface crc and               if we wait, depending on e.g. compiler defines they may get a               different defid (e.g. when a function is called, its procdef is               registered, so depending on whether or not, or when, an interface               procedure is called in the implementation, that may change its               defid otherwise) }             if assigned(current_module) and                current_module.in_interface then               register_def             else               maybe_put_in_symtable_stack;           end;      end;    destructor tstoreddef.destroy;      var        i : longint;      begin        { Direct calls are not allowed, use symtable.deletedef() }        if assigned(owner) then          internalerror(200612311);        if assigned(generictokenbuf) then          begin            generictokenbuf.free;            generictokenbuf:=nil;          end;        genericparas.free;        if assigned(genericparaderefs) then          for i:=0 to genericparaderefs.count-1 do            dispose(pderef(genericparaderefs[i]));        genericparaderefs.free;        genconstraintdata.free;{$ifndef symansistr}        stringdispose(_fullownerhierarchyname);{$endif not symansistr}        inherited destroy;      end;    constructor tstoreddef.ppuload(dt:tdeftyp;ppufile:tcompilerppufile);      var        sizeleft,i,cnt : longint;        buf  : array[0..255] of byte;        symderef : pderef;      begin         inherited create(dt);         DefId:=ppufile.getlongint;         current_module.deflist[DefId]:=self;{$ifdef EXTDEBUG}         fillchar(fileinfo,sizeof(fileinfo),0);{$endif}         { load }         ppufile.getderef(typesymderef);         ppufile.getsmallset(defoptions);         ppufile.getsmallset(defstates);         if df_genconstraint in defoptions then           begin             genconstraintdata:=tgenericconstraintdata.create;             genconstraintdata.ppuload(ppufile);           end;         if [df_generic,df_specialization]*defoptions<>[] then           begin             cnt:=ppufile.getlongint;             if cnt>0 then               begin                 genericparas:=tfphashobjectlist.create(false);                 genericparaderefs:=tfplist.create;                 for i:=0 to cnt-1 do                   begin                     genericparas.add(ppufile.getstring,nil);                     New(symderef);                     ppufile.getderef(symderef^);                     genericparaderefs.add(symderef);                   end;               end;           end;         if df_generic in defoptions then           begin             sizeleft:=ppufile.getlongint;             initgeneric;             while sizeleft>0 do               begin                 if sizeleft>sizeof(buf) then                   i:=sizeof(buf)                 else                   i:=sizeleft;                 ppufile.getdata(buf,i);                 generictokenbuf.write(buf,i);                 dec(sizeleft,i);               end;           end;        if df_specialization in defoptions then          ppufile.getderef(genericdefderef);      end;    function tstoreddef.needs_separate_initrtti:boolean;      begin        result:=false;      end;    function tstoreddef.rtti_mangledname(rt : trttitype) : TSymStr;      var        prefix : string[4];      begin        if (rt=fullrtti) or (not needs_separate_initrtti) then          begin            prefix:='RTTI';            include(defstates,ds_rtti_table_used);          end        else          begin            prefix:='INIT';            include(defstates,ds_init_table_used);          end;        if assigned(typesym) and           (owner.symtabletype in [staticsymtable,globalsymtable]) then          result:=make_mangledname(prefix,typesym.owner,typesym.name)        else          result:=make_mangledname(prefix,findunitsymtable(owner),'def'+unique_id_str)      end;    function tstoreddef.OwnerHierarchyName: string;      var        tmp: tdef;      begin        tmp:=self;        result:='';        repeat          { can be not assigned in case of a forwarddef }          if assigned(tmp.owner) and             (tmp.owner.symtabletype in [ObjectSymtable,recordsymtable]) then            tmp:=tdef(tmp.owner.defowner)          else            break;          result:=tabstractrecorddef(tmp).objrealname^+'.'+result;        until tmp=nil;      end;    function tstoreddef.fullownerhierarchyname(skipprocparams:boolean): TSymStr;      var        lastowner: tsymtable;        tmp: tdef;        pno: tprocnameoptions;      begin{$ifdef symansistr}        if not skipprocparams and (_fullownerhierarchyname<>'') then          exit(_fullownerhierarchyname);{$else symansistr}        if not skipprocparams and assigned(_fullownerhierarchyname) then          exit(_fullownerhierarchyname^);{$endif symansistr}        { the def can only reside inside structured types or          procedures/functions/methods }        tmp:=self;        result:='';        repeat          lastowner:=tmp.owner;          { can be not assigned in case of a forwarddef }          if not assigned(lastowner) then            break          else            tmp:=tdef(lastowner.defowner);          if not assigned(tmp) then            break;          if tmp.typ in [recorddef,objectdef] then            result:=tabstractrecorddef(tmp).objrealname^+'.'+result          else            if tmp.typ=procdef then              begin                pno:=[pno_paranames,pno_proctypeoption];                if skipprocparams then                  include(pno,pno_noparams);                result:=tprocdef(tmp).customprocname(pno)+'.'+result;              end;        until tmp=nil;        { add the unit name }        if assigned(lastowner) and           assigned(lastowner.realname) then          result:=lastowner.realname^+'.'+result;        if not skipprocparams then          { don't store the name in this case }{$ifdef symansistr}          _fullownerhierarchyname:=result;{$else symansistr}          _fullownerhierarchyname:=stringdup(result);{$endif symansistr}      end;    function tstoreddef.in_currentunit: boolean;      var        st: tsymtable;      begin        st:=owner;        while not(st.symtabletype in [globalsymtable,staticsymtable]) do          st:=st.defowner.owner;        result:=st.iscurrentunit;      end;    function tstoreddef.getcopy : tstoreddef;      begin        Message(sym_e_cant_create_unique_type);        getcopy:=cerrordef.create;      end;    procedure tstoreddef.ppuwrite(ppufile:tcompilerppufile);      var        sizeleft,i : longint;        buf  : array[0..255] of byte;        oldintfcrc : boolean;      begin        if defid<0 then          internalerror(2015101401);        ppufile.putlongint(DefId);        ppufile.putderef(typesymderef);        ppufile.putsmallset(defoptions);        oldintfcrc:=ppufile.do_crc;        ppufile.do_crc:=false;        ppufile.putsmallset(defstates);        if df_genconstraint in defoptions then          genconstraintdata.ppuwrite(ppufile);        if [df_generic,df_specialization]*defoptions<>[] then          begin            if not assigned(genericparas) then              ppufile.putlongint(0)            else              begin                if not assigned(genericparaderefs) then                  internalerror(2014052305);                ppufile.putlongint(genericparas.count);                for i:=0 to genericparas.count-1 do                  begin                    ppufile.putstring(genericparas.nameofindex(i));                    ppufile.putderef(pderef(genericparaderefs[i])^);                  end;              end;          end;        if df_generic in defoptions then          begin            if assigned(generictokenbuf) then              begin                sizeleft:=generictokenbuf.size;                generictokenbuf.seek(0);              end            else              sizeleft:=0;            ppufile.putlongint(sizeleft);            while sizeleft>0 do              begin                if sizeleft>sizeof(buf) then                  i:=sizeof(buf)                else                  i:=sizeleft;                generictokenbuf.read(buf,i);                ppufile.putdata(buf,i);                dec(sizeleft,i);              end;          end;        ppufile.do_crc:=oldintfcrc;        if df_specialization in defoptions then          ppufile.putderef(genericdefderef);      end;    procedure tstoreddef.buildderef;      var        i : longint;        sym : tsym;        symderef : pderef;      begin        if not registered then          register_def;        typesymderef.build(typesym);        genericdefderef.build(genericdef);        if assigned(genconstraintdata) then          genconstraintdata.buildderef;        if assigned(genericparas) then          begin            if not assigned(genericparaderefs) then              genericparaderefs:=tfplist.create;            for i:=0 to genericparas.count-1 do              begin                sym:=tsym(genericparas.items[i]);                new(symderef);                symderef^.build(sym);                genericparaderefs.add(symderef);              end;          end;      end;    procedure tstoreddef.buildderefimpl;      begin      end;    procedure tstoreddef.deref;      var        symderef : pderef;        i : longint;      begin        typesym:=ttypesym(typesymderef.resolve);        if df_specialization in defoptions then          genericdef:=tstoreddef(genericdefderef.resolve);        if assigned(genconstraintdata) then          genconstraintdata.deref;        if assigned(genericparas) then          begin            if not assigned(genericparaderefs) then              internalerror(2014052302);            if genericparas.count<>genericparaderefs.count then              internalerror(2014052303);            for i:=0 to genericparaderefs.count-1 do              begin                symderef:=pderef(genericparaderefs[i]);                genericparas.items[i]:=symderef^.resolve;              end;          end;      end;    procedure tstoreddef.derefimpl;      begin      end;    function tstoreddef.size : asizeint;      begin         size:=savesize;      end;    function tstoreddef.getvardef:longint;      begin        result:=varUndefined;      end;    function tstoreddef.alignment : shortint;      begin         { natural alignment by default }         alignment:=size_2_align(savesize);         { can happen if savesize = 0, e.g. for voiddef or           an empty record         }         if (alignment=0) then           alignment:=1;      end;    { returns true, if the definition can be published }    function tstoreddef.is_publishable : boolean;      begin         is_publishable:=false;      end;    { needs an init table }    function tstoreddef.needs_inittable : boolean;      begin         needs_inittable:=false;      end;    function tstoreddef.has_non_trivial_init_child(check_parent:boolean):boolean;      begin        result:=false;      end;   function tstoreddef.is_intregable : boolean;     var       recsize,temp: longint;     begin        case typ of          orddef,          pointerdef,          enumdef,          classrefdef:            is_intregable:=true;          procvardef :            is_intregable:=tprocvardef(self).is_addressonly or (po_methodpointer in tprocvardef(self).procoptions);          objectdef:            is_intregable:=(is_implicit_pointer_object_type(self)) and not needs_inittable;          setdef:            is_intregable:=is_smallset(self);          arraydef:{$ifdef cpuhighleveltarget}            is_intregable:=false{$else cpuhighleveltarget}            is_intregable:=not(is_special_array(self)) and              (tarraydef(self).size in [1,2,4,8]) and tstoreddef(tarraydef(self).elementdef).is_intregable{$ifdef SUPPORT_MMX}              and not((cs_mmx in current_settings.localswitches) and                 is_mmx_able_array(self)){$endif SUPPORT_MMX}{$endif cpuhighleveltarget}              ;          recorddef:            begin{$ifdef cpuhighleveltarget}              is_intregable:=false;{$else cpuhighleveltarget}              recsize:=size;              is_intregable:=                ispowerof2(recsize,temp) and                ((recsize<=sizeof(aint)*2) and                 not trecorddef(self).contains_cross_aword_field and                 { records cannot go into registers on 16 bit targets for now }                 (sizeof(aint)>2) and                 (not trecorddef(self).contains_float_field) or                  (recsize <= sizeof(aint))                 ) and                 not needs_inittable;{$endif cpuhighleveltarget}            end;          else           is_intregable:=false;        end;     end;   function tstoreddef.is_fpuregable : boolean;     begin{$ifdef x86}       result:=use_vectorfpu(self);{$else x86}       result:=(typ=floatdef) and not(cs_fp_emulation in current_settings.moduleswitches);{$endif x86}     end;   function tstoreddef.is_const_intregable : boolean;     begin       case typ of         stringdef:           result:=tstringdef(self).stringtype in [st_ansistring,st_unicodestring,st_widestring];         arraydef:           result:=is_dynamic_array(self);         objectdef:           result:=is_interface(self);         else           result:=false;       end;     end;   procedure tstoreddef.initgeneric;     begin       if assigned(generictokenbuf) then         internalerror(200512131);       generictokenbuf:=tdynamicarray.create(256);     end;   function tstoreddef.is_generic: boolean;     var       sym: tsym;       i: longint;     begin       result:=assigned(genericparas) and                 (genericparas.count>0) and                 (df_generic in defoptions);       if result then         { if any of the type parameters does *not* belong to as (meaning it was passed           in from outside) then we aren't a generic, but a specialization }         for i:=0 to genericparas.count-1 do           begin             sym:=tsym(genericparas[i]);             if sym.typ<>symconst.typesym then               internalerror(2014050903);             if sym.owner.defowner<>self then               exit(false);           end;     end;   function tstoreddef.is_specialization: boolean;     var       i : longint;       sym : tsym;     begin       result:=assigned(genericparas) and                 (genericparas.count>0) and                 (df_specialization in defoptions);       if result then         begin           { if at least one of the generic parameters is not owned by us (meaning it was             passed in from outside) then we have a specialization, otherwise we have a generic }           for i:=0 to genericparas.count-1 do             begin               sym:=tsym(genericparas[i]);               if sym.typ<>symconst.typesym then                 internalerror(2014050904);               if sym.owner.defowner<>self then                 exit(true);             end;           result:=false;         end;     end;   procedure tstoreddef.register_def;     var       gst : tgetsymtable;       st : tsymtable;     begin       if registered then         exit;       { Register in current_module }       if assigned(current_module) then         begin           exclude(defoptions,df_not_registered_no_free);           for gst:=low(tgetsymtable) to high(tgetsymtable) do             begin               st:=getsymtable(gst);               if assigned(st) then                 tstoredsymtable(st).register_children;             end;           if defid<defid_not_registered then             defid:=deflist_index           else             begin               current_module.deflist.Add(self);               defid:=current_module.deflist.Count-1;             end;           maybe_put_in_symtable_stack;         end       else         DefId:=defid_registered_nost;     end;   procedure tstoreddef.maybe_put_in_symtable_stack;     var       insertstack: psymtablestackitem;     begin       if assigned(symtablestack) and          not assigned(self.owner) then         begin           insertstack:=symtablestack.stack;           { don't insert defs in exception symtables, as they are freed before             the module is compiled, so we can get crashes on high level targets             if they still need it while e.g. writing assembler code }           while assigned(insertstack) and                 (insertstack^.symtable.symtabletype in [stt_exceptsymtable,withsymtable]) do             insertstack:=insertstack^.next;           if not assigned(insertstack) then             internalerror(200602044);           if insertstack^.symtable.sealed then             internalerror(2015022301);           insertstack^.symtable.insertdef(self);         end;     end;{****************************************************************************                               Tstringdef****************************************************************************}    constructor tstringdef.createshort(l: byte; doregister: boolean);      begin         inherited create(stringdef,doregister);         stringtype:=st_shortstring;         encoding:=0;         len:=l;      end;    constructor tstringdef.loadshort(ppufile:tcompilerppufile);      begin         inherited ppuload(stringdef,ppufile);         stringtype:=st_shortstring;         encoding:=0;         len:=ppufile.getbyte;         ppuload_platform(ppufile);      end;    constructor tstringdef.createlong(l: asizeint; doregister: boolean);      begin         inherited create(stringdef,doregister);         stringtype:=st_longstring;         encoding:=0;         len:=l;      end;    constructor tstringdef.loadlong(ppufile:tcompilerppufile);      begin         inherited ppuload(stringdef,ppufile);         stringtype:=st_longstring;         encoding:=0;         len:=ppufile.getasizeint;         ppuload_platform(ppufile);      end;    constructor tstringdef.createansi(aencoding: tstringencoding; doregister: boolean);      begin         inherited create(stringdef,doregister);         stringtype:=st_ansistring;         encoding:=aencoding;         len:=-1;      end;    constructor tstringdef.loadansi(ppufile:tcompilerppufile);      begin         inherited ppuload(stringdef,ppufile);         stringtype:=st_ansistring;         len:=ppufile.getasizeint;         encoding:=ppufile.getword;         ppuload_platform(ppufile);      end;    constructor tstringdef.createwide(doregister: boolean);      begin         inherited create(stringdef,doregister);         stringtype:=st_widestring;         if target_info.endian=endian_little then           encoding:=CP_UTF16LE         else           encoding:=CP_UTF16BE;         len:=-1;      end;    constructor tstringdef.loadwide(ppufile:tcompilerppufile);      begin         inherited ppuload(stringdef,ppufile);         stringtype:=st_widestring;         if target_info.endian=endian_little then           encoding:=CP_UTF16LE         else           encoding:=CP_UTF16BE;         len:=ppufile.getasizeint;         ppuload_platform(ppufile);      end;    constructor tstringdef.createunicode(doregister: boolean);      begin         inherited create(stringdef,doregister);         stringtype:=st_unicodestring;         if target_info.endian=endian_little then           encoding:=CP_UTF16LE         else           encoding:=CP_UTF16BE;         len:=-1;      end;    constructor tstringdef.loadunicode(ppufile:tcompilerppufile);      begin         inherited ppuload(stringdef,ppufile);         stringtype:=st_unicodestring;         len:=ppufile.getasizeint;         encoding:=ppufile.getword;         ppuload_platform(ppufile);      end;    function tstringdef.getcopy : tstoreddef;      begin        result:=cstringdef.create(typ,true);        result.typ:=stringdef;        tstringdef(result).stringtype:=stringtype;        tstringdef(result).encoding:=encoding;        tstringdef(result).len:=len;      end;    function tstringdef.stringtypname:string;      const        typname:array[tstringtype] of string[10]=(          'shortstr','longstr','ansistr','widestr','unicodestr'        );      begin        stringtypname:=typname[stringtype];      end;    procedure tstringdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         if stringtype=st_shortstring then           begin{$ifdef extdebug}            if len > 255 then internalerror(12122002);{$endif}            ppufile.putbyte(byte(len))           end         else           ppufile.putasizeint(len);         if stringtype in [st_ansistring,st_unicodestring] then           ppufile.putword(encoding);         case stringtype of            st_shortstring : writeentry(ppufile,ibshortstringdef);            st_longstring : writeentry(ppufile,iblongstringdef);            st_ansistring : writeentry(ppufile,ibansistringdef);            st_widestring : writeentry(ppufile,ibwidestringdef);            st_unicodestring : writeentry(ppufile,ibunicodestringdef);         end;      end;    function tstringdef.needs_inittable : boolean;      begin         needs_inittable:=stringtype in [st_ansistring,st_widestring,st_unicodestring];      end;    function tstringdef.GetTypeName : string;      const         names : array[tstringtype] of string[15] = (           'ShortString','LongString','AnsiString','WideString','UnicodeString');      begin         GetTypeName:=names[stringtype];      end;    function tstringdef.getvardef : longint;      const        vardef : array[tstringtype] of longint = (          varUndefined,varUndefined,varString,varOleStr,varUString);      begin        result:=vardef[stringtype];      end;    function tstringdef.alignment : shortint;      begin        case stringtype of          st_unicodestring,          st_widestring,          st_ansistring:            alignment:=voidpointertype.alignment;          st_longstring,          st_shortstring:              { char to string accesses byte 0 and 1 with one word access }            if (tf_requires_proper_alignment in target_info.flags) or              { macpas needs an alignment of 2 (MetroWerks compatible) }               (m_mac in current_settings.modeswitches) then              alignment:=size_2_align(2)            else              alignment:=size_2_align(1);        end;      end;    function tstringdef.getmangledparaname : TSymStr;      begin        getmangledparaname:='STRING';      end;    function tstringdef.is_publishable : boolean;      begin         is_publishable:=true;      end;    function tstringdef.size: asizeint;      begin        case stringtype of          st_shortstring:            Result:=len+1;          st_longstring,          st_ansistring,          st_widestring,          st_unicodestring:            Result:=voidpointertype.size;        end;      end;{****************************************************************************                                 TENUMDEF****************************************************************************}    constructor tenumdef.create;      begin         inherited create(enumdef,true);         minval:=0;         maxval:=0;         calcsavesize(current_settings.packenum);         has_jumps:=false;         basedef:=nil;         basedefderef.reset;         symtable:=tenumsymtable.create(self);      end;    constructor tenumdef.create_subrange(_basedef:tenumdef;_min,_max:asizeint);      begin         inherited create(enumdef,true);         minval:=_min;         maxval:=_max;         basedef:=_basedef;         calcsavesize(current_settings.packenum);         has_jumps:=false;         symtable:=basedef.symtable.getcopy;         include(defoptions, df_copied_def);       end;    constructor tenumdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(enumdef,ppufile);         minval:=ppufile.getaint;         maxval:=ppufile.getaint;         savesize:=ppufile.getaint;         has_jumps:=false;         if df_copied_def in defoptions then           begin             symtable:=nil;             ppufile.getderef(basedefderef);             ppuload_platform(ppufile);           end         else           begin             ppuload_platform(ppufile);             // create with nil defowner first to prevent values changes on insert             symtable:=tenumsymtable.create(nil);             tenumsymtable(symtable).ppuload(ppufile);             symtable.defowner:=self;           end;      end;    destructor tenumdef.destroy;      begin        symtable.free;        symtable:=nil;        inherited destroy;      end;    function tenumdef.getcopy : tstoreddef;      begin        if assigned(basedef) then          result:=cenumdef.create_subrange(basedef,minval,maxval)        else          begin            result:=cenumdef.create;            tenumdef(result).minval:=minval;            tenumdef(result).maxval:=maxval;            tenumdef(result).symtable.free;            tenumdef(result).symtable:=symtable.getcopy;            tenumdef(result).basedef:=self;          end;        tenumdef(result).has_jumps:=has_jumps;        tenumdef(result).basedefderef:=basedefderef;        include(tenumdef(result).defoptions,df_copied_def);      end;    procedure tenumdef.calcsavesize(packenum: shortint);      begin{$IFNDEF cpu64bitaddr} {$push}{$warnings off} {$ENDIF} //comparison always false warning        if (packenum=8) or (int64(min)<low(longint)) or (int64(max)>high(cardinal)) then         savesize:=8{$IFNDEF cpu64bitaddr} {$pop} {$ENDIF}        else{$IFDEF cpu16bitaddr} {$push}{$warnings off} {$ENDIF} //comparison always false warning         if (packenum=4) or (min<low(smallint)) or (max>high(word)) then          savesize:=4{$IFDEF cpu16bitaddr} {$pop} {$ENDIF}        else         if (packenum=2) or (min<low(shortint)) or (max>high(byte)) then          savesize:=2        else         savesize:=1;      end;    function tenumdef.packedbitsize: asizeint;      var        sizeval: tconstexprint;        power: longint;      begin        result := 0;        if (minval >= 0) and           (maxval <= 1) then          result := 1        else          begin            if (minval>=0) then              sizeval:=maxval            else             { don't count 0 twice, but take into account that range goes from -n-1..n }              sizeval:=(cutils.max(-minval,maxval+1)*2)-1;            { 256 must become 512 etc. }            nextpowerof2(sizeval+1,power);            result := power;          end;      end;    procedure tenumdef.setmax(_max:asizeint);      begin        maxval:=_max;        calcsavesize(current_settings.packenum);      end;    procedure tenumdef.setmin(_min:asizeint);      begin        minval:=_min;        calcsavesize(current_settings.packenum);      end;    function tenumdef.min:asizeint;      begin        min:=minval;      end;    function tenumdef.max:asizeint;      begin        max:=maxval;      end;    function tenumdef.getfirstsym: tsym;      var        i:integer;      begin        for i := 0 to symtable.SymList.Count - 1 do          begin            result:=tsym(symtable.SymList[i]);            if tenumsym(result).value=minval then              exit;          end;        result:=nil;      end;    function tenumdef.int2enumsym(l: asizeint): tsym;      var        i: longint;        sym: tsym;        bdef: tenumdef;      begin        result:=nil;        if (l<minval) or           (l>maxval) then          exit;        bdef:=getbasedef;        for i:=0 to bdef.symtable.symlist.count-1 do          begin            sym:=tsym(bdef.symtable.symlist[i]);            if (sym.typ=enumsym) and               (tenumsym(sym).value=l) then              begin                result:=sym;                exit;              end;          end;      end;    function tenumdef.getbasedef: tenumdef;      begin        if not assigned(basedef) then          result:=self        else          result:=basedef;      end;    procedure tenumdef.buildderef;      begin        inherited buildderef;        if df_copied_def in defoptions then          basedefderef.build(basedef)        else          tenumsymtable(symtable).buildderef;      end;    procedure tenumdef.deref;      begin        inherited deref;        if df_copied_def in defoptions then          begin            basedef:=tenumdef(basedefderef.resolve);            symtable:=basedef.symtable.getcopy;          end        else          tenumsymtable(symtable).deref(false);      end;    procedure tenumdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putaint(min);         ppufile.putaint(max);         ppufile.putaint(savesize);         if df_copied_def in defoptions then           ppufile.putderef(basedefderef);         writeentry(ppufile,ibenumdef);         if not (df_copied_def in defoptions) then           tenumsymtable(symtable).ppuwrite(ppufile);      end;    function tenumdef.is_publishable : boolean;      begin         is_publishable:=true;      end;    function tenumdef.GetTypeName : string;      begin         GetTypeName:='<enumeration type>';      end;{****************************************************************************                                 TORDDEF****************************************************************************}    constructor torddef.create(t : tordtype;v,b : TConstExprInt; doregister: boolean);      begin         inherited create(orddef,doregister);         low:=v;         high:=b;         ordtype:=t;         setsize;      end;    constructor torddef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(orddef,ppufile);         ordtype:=tordtype(ppufile.getbyte);         low:=ppufile.getexprint;         high:=ppufile.getexprint;         setsize;         ppuload_platform(ppufile);      end;    function torddef.getcopy : tstoreddef;      begin         result:=corddef.create(ordtype,low,high,true);         result.typ:=orddef;         torddef(result).low:=low;         torddef(result).high:=high;         torddef(result).ordtype:=ordtype;         torddef(result).savesize:=savesize;      end;    function torddef.alignment:shortint;      begin        if (target_info.system in [system_i386_darwin,system_i386_iphonesim,system_arm_darwin]) and           (ordtype in [s64bit,u64bit]) then          result := 4        else          result := inherited alignment;      end;    procedure torddef.setsize;      const        sizetbl : array[tordtype] of longint = (          0,          1,2,4,8,16,          1,2,4,8,16,          1,1,2,4,8,          1,2,4,8,          1,2,8,system.high(longint)        );      begin        savesize:=sizetbl[ordtype];        if savesize=system.high(longint) then          savesize:=packedbitsize div 8;      end;    function torddef.packedbitsize: asizeint;      var        sizeval: tconstexprint;        power: longint;      begin        result := 0;        if ordtype = uvoid then          exit;{$ifndef cpu64bitalu}        if (ordtype in [s64bit,u64bit]) then{$else not cpu64bitalu}        if (ordtype = u64bit) or           ((ordtype = s64bit) and            ((low <= (system.low(int64) div 2)) or             (high > (system.high(int64) div 2)))) then{$endif cpu64bitalu}          result := 64        else if (            (low >= 0) and            (high <= 1)           ) or (             ordtype in [pasbool1,pasbool8,pasbool16,pasbool32,pasbool64,bool8bit,bool16bit,bool32bit,bool64bit]           ) then          result := 1        else          begin            if (low>=0) then              sizeval:=high            else              { don't count 0 twice, but take into account that range goes from -n-1..n }              sizeval:=(cutils.max(-low,high+1)*2)-1;            { 256 must become 512 etc. }            nextpowerof2(sizeval+1,power);            result := power;         end;      end;    function torddef.getvardef : longint;      const        basetype2vardef : array[tordtype] of longint = (          varUndefined,          varbyte,varword,varlongword,varqword,varUndefined,          varshortint,varsmallint,varinteger,varint64,varUndefined,          varboolean,varboolean,varboolean,varboolean,varboolean,          varboolean,varboolean,varUndefined,varUndefined,          varUndefined,varUndefined,varCurrency,varEmpty);      begin        result:=basetype2vardef[ordtype];        if result=varEmpty then          result:=basetype2vardef[range_to_basetype(low,high)];      end;    procedure torddef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(ordtype));         ppufile.putexprint(low);         ppufile.putexprint(high);         writeentry(ppufile,iborddef);      end;    function torddef.is_publishable : boolean;      begin         is_publishable:=(ordtype<>uvoid);      end;    function torddef.GetTypeName : string;      const        names : array[tordtype] of string[20] = (          'untyped',          'Byte','Word','DWord','QWord','UInt128',          'ShortInt','SmallInt','LongInt','Int64','Int128',          'Boolean','Boolean8','Boolean16','Boolean32','Boolean64',          'ByteBool','WordBool','LongBool','QWordBool',          'Char','WideChar','Currency','CustomRange');      begin         GetTypeName:=names[ordtype];      end;{****************************************************************************                                TFLOATDEF****************************************************************************}    constructor tfloatdef.create(t: tfloattype; doregister: boolean);      begin         inherited create(floatdef,doregister);         floattype:=t;         setsize;      end;    constructor tfloatdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(floatdef,ppufile);         floattype:=tfloattype(ppufile.getbyte);         setsize;         ppuload_platform(ppufile);      end;    function tfloatdef.getcopy : tstoreddef;      begin         result:=cfloatdef.create(floattype,true);         result.typ:=floatdef;         tfloatdef(result).savesize:=savesize;      end;    function tfloatdef.alignment:shortint;      begin        if (target_info.system in [system_i386_darwin,system_i386_iphonesim,system_arm_darwin]) then          case floattype of            sc80real,            s80real: result:=16;            s64real,            s64currency,            s64comp : result:=4;            else              result := inherited alignment;          end        else          result := inherited alignment;      end;    function tfloatdef.structalignment: shortint;      begin        { aix is really annoying: the recommended scalar alignment for both          int64 and double is 64 bits, but in structs int64 has to be aligned          to 8 bytes and double to 4 bytes }        if (target_info.system in systems_aix) and           (floattype=s64real) then          result:=4        else          result:=alignment;      end;    procedure tfloatdef.setsize;      begin         case floattype of           s32real : savesize:=4;           s80real : savesize:=10;           sc80real:             if target_info.system in [system_i386_darwin,                  system_i386_iphonesim,system_x86_64_darwin,                  system_x86_64_iphonesim,                  system_x86_64_linux,system_x86_64_freebsd,                  system_x86_64_openbsd,system_x86_64_netbsd,                  system_x86_64_solaris,system_x86_64_embedded,                  system_x86_64_dragonfly,system_x86_64_haiku] then               savesize:=16             else               savesize:=12;           s64real,           s64currency,           s64comp : savesize:=8;         else           savesize:=0;         end;      end;    function tfloatdef.getvardef : longint;      const        floattype2vardef : array[tfloattype] of longint = (          varSingle,varDouble,varUndefined,varUndefined,          varUndefined,varCurrency,varUndefined);      begin        if (upper(typename)='TDATETIME') and          assigned(owner) and          assigned(owner.name) and          (owner.name^='SYSTEM') then          result:=varDate        else          result:=floattype2vardef[floattype];      end;    procedure tfloatdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(floattype));         writeentry(ppufile,ibfloatdef);      end;    function tfloatdef.is_publishable : boolean;      begin         is_publishable:=true;      end;    function tfloatdef.GetTypeName : string;      const        names : array[tfloattype] of string[20] = (          'Single','Double','Extended','CExtended','Comp','Currency','Float128');      begin         GetTypeName:=names[floattype];      end;{****************************************************************************                                TFILEDEF****************************************************************************}    constructor tfiledef.createtext;      begin         inherited create(filedef,true);         filetyp:=ft_text;         typedfiledef:=nil;         typedfiledefderef.reset;      end;    constructor tfiledef.createuntyped;      begin         inherited create(filedef,true);         filetyp:=ft_untyped;         typedfiledef:=nil;      end;    constructor tfiledef.createtyped(def:tdef);      begin         inherited create(filedef,true);         filetyp:=ft_typed;         typedfiledef:=def;      end;    constructor tfiledef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(filedef,ppufile);         filetyp:=tfiletyp(ppufile.getbyte);         if filetyp=ft_typed then           ppufile.getderef(typedfiledefderef)         else           typedfiledef:=nil;         ppuload_platform(ppufile);      end;    function tfiledef.getcopy : tstoreddef;      begin        case filetyp of          ft_typed:            result:=cfiledef.createtyped(typedfiledef);          ft_untyped:            result:=cfiledef.createuntyped;          ft_text:            result:=cfiledef.createtext;        end;      end;    procedure tfiledef.buildderef;      begin        inherited buildderef;        if filetyp=ft_typed then          typedfiledefderef.build(typedfiledef);      end;    procedure tfiledef.deref;      begin        inherited deref;        if filetyp=ft_typed then          typedfiledef:=tdef(typedfiledefderef.resolve);      end;    function  tfiledef.size:asizeint;      begin        if savesize=0 then          setsize;        size:=savesize;      end;    procedure tfiledef.setsize;      begin       case filetyp of         ft_text:           savesize:=search_system_type('TEXTREC').typedef.size;         ft_typed:           begin             savesize:=search_system_type('FILEREC').typedef.size;             { allocate put/get buffer in iso mode }             if m_isolike_io in current_settings.modeswitches then               inc(savesize,typedfiledef.size);           end;         ft_untyped:           savesize:=search_system_type('FILEREC').typedef.size;         end;      end;    function tfiledef.alignment: shortint;      begin        case filetyp of          ft_text:            result:=search_system_type('TEXTREC').typedef.alignment;          ft_typed,          ft_untyped:            result:=search_system_type('FILEREC').typedef.alignment;          end;      end;    procedure tfiledef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(filetyp));         if filetyp=ft_typed then           ppufile.putderef(typedfiledefderef);         writeentry(ppufile,ibfiledef);      end;    function tfiledef.GetTypeName : string;      begin         case filetyp of           ft_untyped:             GetTypeName:='File';           ft_typed:             GetTypeName:='File Of '+typedfiledef.typename;           ft_text:             GetTypeName:='Text'         end;      end;    function tfiledef.getmangledparaname : TSymStr;      begin         case filetyp of           ft_untyped:             getmangledparaname:='FILE';           ft_typed:             getmangledparaname:='FILE$OF$'+typedfiledef.mangledparaname;           ft_text:             getmangledparaname:='TEXT'         end;      end;{****************************************************************************                               TVARIANTDEF****************************************************************************}    constructor tvariantdef.create(v : tvarianttype);      begin         inherited create(variantdef,true);         varianttype:=v;         setsize;      end;    constructor tvariantdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(variantdef,ppufile);         varianttype:=tvarianttype(ppufile.getbyte);         setsize;         ppuload_platform(ppufile);      end;    function tvariantdef.getcopy : tstoreddef;      begin        result:=cvariantdef.create(varianttype);      end;    procedure tvariantdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(varianttype));         writeentry(ppufile,ibvariantdef);      end;    function tvariantdef.getvardef : longint;      begin        Result:=varVariant;      end;    procedure tvariantdef.setsize;      begin{$ifdef cpu64bitaddr}        savesize:=24;{$else cpu64bitaddr}        savesize:=16;{$endif cpu64bitaddr}      end;    function tvariantdef.GetTypeName : string;      begin         case varianttype of           vt_normalvariant:             GetTypeName:='Variant';           vt_olevariant:             GetTypeName:='OleVariant';         end;      end;    function tvariantdef.needs_inittable : boolean;      begin         needs_inittable:=true;      end;    function tvariantdef.is_publishable : boolean;      begin         is_publishable:=true;      end;{****************************************************************************                            TABSTRACtpointerdef****************************************************************************}    constructor tabstractpointerdef.create(dt:tdeftyp;def:tdef);      begin        inherited create(dt,true);        pointeddef:=def;        pointeddefderef.reset;        if df_generic in pointeddef.defoptions then          include(defoptions,df_generic);        if df_specialization in pointeddef.defoptions then          include(defoptions,df_specialization);      end;    constructor tabstractpointerdef.ppuload(dt:tdeftyp;ppufile:tcompilerppufile);      begin         inherited ppuload(dt,ppufile);         ppufile.getderef(pointeddefderef);      end;    procedure tabstractpointerdef.buildderef;      begin        inherited buildderef;        pointeddefderef.build(pointeddef);      end;    procedure tabstractpointerdef.deref;      begin        inherited deref;        pointeddef:=tdef(pointeddefderef.resolve);      end;    function tabstractpointerdef.size: asizeint;      begin        Result:=voidpointertype.size;      end;    function tabstractpointerdef.alignment: shortint;      begin        alignment:=size_2_align(voidpointertype.size);      end;    procedure tabstractpointerdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putderef(pointeddefderef);      end;{****************************************************************************                               tpointerdef****************************************************************************}    constructor tpointerdef.create(def:tdef);      begin        inherited create(pointerdef,def);        has_pointer_math:=cs_pointermath in current_settings.localswitches;        if (df_specialization in tstoreddef(def).defoptions){$ifndef genericdef_for_nested}           { currently, nested procdefs of generic routines get df_specialisation,             but no genericdef }           and assigned(tstoreddef(def).genericdef){$endif}           then          genericdef:=cpointerdef.getreusable(tstoreddef(def).genericdef);      end;    class function tpointerdef.getreusable(def: tdef): tpointerdef;      var        res: PHashSetItem;        oldsymtablestack: tsymtablestack;      begin        if not assigned(current_module) then          internalerror(2011071101);        res:=current_module.ptrdefs.FindOrAdd(@def,sizeof(def));        if not assigned(res^.Data) then          begin            { since these pointerdefs can be reused anywhere in the current              unit, add them to the global/staticsymtable (or local symtable              if they're a local def, because otherwise they'll be saved              to the ppu referencing a local symtable entry that doesn't              exist in the ppu) }            oldsymtablestack:=symtablestack;            { do not simply push/pop current_module.localsymtable, because              that can have side-effects (e.g., it removes helpers) }            symtablestack:=nil;            result:=cpointerdef.create(def);            setup_reusable_def(def,result,res,oldsymtablestack);            { res^.Data may still be nil -> don't overwrite result }            exit;          end;        result:=tpointerdef(res^.Data);      end;    class function tpointerdef.getreusable_no_free(def: tdef): tpointerdef;      begin        result:=getreusable(def);        if not result.is_registered then          include(result.defoptions,df_not_registered_no_free);      end;    function tpointerdef.size: asizeint;      begin        result:=sizeof(pint);      end;    constructor tpointerdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(pointerdef,ppufile);         has_pointer_math:=(ppufile.getbyte<>0);         ppuload_platform(ppufile);      end;    function tpointerdef.getcopy : tstoreddef;      begin        { don't use direct pointeddef if it is a forwarddef because in other case          one of them will be destroyed on forward type resolve and the second will          point to garbage }        if pointeddef.typ=forwarddef then          result:=cpointerdef.create(tforwarddef(pointeddef).getcopy)        else          result:=cpointerdef.create(pointeddef);        tpointerdef(result).has_pointer_math:=has_pointer_math;        tpointerdef(result).savesize:=savesize;      end;    procedure tpointerdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(has_pointer_math));         writeentry(ppufile,ibpointerdef);      end;    function tpointerdef.GetTypeName : string;      begin        { parameter types and the resultdef of a procvardef can contain a          pointer to this procvardef itself, resulting in endless recursion ->          use the typesym's name instead if it exists (if it doesn't, such as          for anynonymous procedure types in macpas/iso mode, then there cannot          be any recursive references to it either) }        if (pointeddef.typ<>procvardef) or           not assigned(pointeddef.typesym) then          GetTypeName:='^'+pointeddef.typename        else          GetTypeName:='^'+pointeddef.typesym.realname;      end;    function tpointerdef.pointer_arithmetic_int_type:tdef;      begin        result:=ptrsinttype;      end;    function tpointerdef.pointer_arithmetic_uint_type:tdef;      begin        result:=ptruinttype;      end;    function tpointerdef.pointer_subtraction_result_type:tdef;      begin        result:=ptrsinttype;      end;    function tpointerdef.compatible_with_pointerdef_size(ptr: tpointerdef): boolean;      begin        result:=true;      end;    function tpointerdef.converted_pointer_to_array_range_type:tdef;      begin        result:=ptrsinttype;      end;{****************************************************************************                              TCLASSREFDEF****************************************************************************}    constructor tclassrefdef.create(def:tdef);      begin         inherited create(classrefdef,def);         if df_specialization in tstoreddef(def).defoptions then           genericdef:=cclassrefdef.create(tstoreddef(def).genericdef);      end;    constructor tclassrefdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(classrefdef,ppufile);         ppuload_platform(ppufile);      end;    procedure tclassrefdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         writeentry(ppufile,ibclassrefdef);      end;    function tclassrefdef.getcopy:tstoreddef;      begin        if pointeddef.typ=forwarddef then          result:=cclassrefdef.create(tforwarddef(pointeddef).getcopy)        else          result:=cclassrefdef.create(pointeddef);      end;    function tclassrefdef.GetTypeName : string;      begin         GetTypeName:='Class Of '+pointeddef.typename;      end;    function tclassrefdef.is_publishable : boolean;      begin         result:=true;      end;    function tclassrefdef.rtti_mangledname(rt: trttitype): TSymStr;      begin        if (tobjectdef(pointeddef).objecttype<>odt_objcclass) then          result:=inherited rtti_mangledname(rt)        else          result:=tobjectdef(pointeddef).rtti_mangledname(objcmetartti);      end;    procedure tclassrefdef.register_created_object_type;      begin        tobjectdef(pointeddef).register_created_classref_type;      end;{***************************************************************************                                   TSETDEF***************************************************************************}    constructor tsetdef.create(def: tdef; low, high: asizeint; doregister: boolean);      var        setallocbits: aint;        packedsavesize: aint;        actual_setalloc: ShortInt;      begin         inherited create(setdef,doregister);         elementdef:=def;         elementdefderef.reset;         setmax:=high;         actual_setalloc:=current_settings.setalloc;{$if defined(cpu8bitalu) or defined(cpu16bitalu)}         if actual_setalloc=0 then           actual_setalloc:=1;{$endif}         if (actual_setalloc=0) then           begin             setbase:=0;             if (high<32) then               savesize:=Sizeof(longint)             else if (high<256) then               savesize:=32             else               savesize:=(high+7) div 8           end         else           begin             setallocbits:=actual_setalloc*8;             setbase:=low and not(setallocbits-1);             packedsavesize:=actual_setalloc*((((high+setallocbits)-setbase)) DIV setallocbits);             savesize:=packedsavesize;{$if not defined(cpu8bitalu) and not defined(cpu16bitalu)}             if savesize=3 then               savesize:=4;{$endif}           end;      end;    constructor tsetdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(setdef,ppufile);         ppufile.getderef(elementdefderef);         savesize:=ppufile.getasizeint;         setbase:=ppufile.getasizeint;         setmax:=ppufile.getasizeint;         ppuload_platform(ppufile);      end;    function tsetdef.getcopy : tstoreddef;      begin        result:=csetdef.create(elementdef,setbase,setmax,true);        { the copy might have been created with a different setalloc setting }        tsetdef(result).savesize:=savesize;      end;    procedure tsetdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putderef(elementdefderef);         ppufile.putasizeint(savesize);         ppufile.putasizeint(setbase);         ppufile.putasizeint(setmax);         writeentry(ppufile,ibsetdef);      end;    procedure tsetdef.buildderef;      begin        inherited buildderef;        elementdefderef.build(elementdef);      end;    procedure tsetdef.deref;      begin        inherited deref;        elementdef:=tdef(elementdefderef.resolve);      end;    function tsetdef.is_publishable : boolean;      begin         is_publishable:=savesize in [1,2,4];      end;    function tsetdef.alignment: shortint;      begin        Result:=inherited;        if result>sizeof(aint) then          result:=sizeof(aint);      end;    function tsetdef.GetTypeName : string;      begin         if assigned(elementdef) then          GetTypeName:='Set Of '+elementdef.typename         else          GetTypeName:='Empty Set';      end;{***************************************************************************                                 TFORMALDEF***************************************************************************}    constructor tformaldef.create(Atyped:boolean);      begin         inherited create(formaldef,true);         typed:=Atyped;         savesize:=0;      end;    constructor tformaldef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(formaldef,ppufile);         typed:=boolean(ppufile.getbyte);         savesize:=0;         ppuload_platform(ppufile);      end;    procedure tformaldef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(typed));         writeentry(ppufile,ibformaldef);      end;    function tformaldef.GetTypeName : string;      begin         if typed then           GetTypeName:='<Typed formal type>'         else           GetTypeName:='<Formal type>';      end;{***************************************************************************                           TARRAYDEF***************************************************************************}    constructor tarraydef.create(l,h:asizeint;def:tdef);      begin         inherited create(arraydef,true);         lowrange:=l;         highrange:=h;         rangedef:=def;         rangedefderef.reset;         _elementdef:=nil;         _elementdefderef.reset;         arrayoptions:=[];         symtable:=tarraysymtable.create(self);      end;    constructor tarraydef.create_openarray;      begin        self.create(0,-1,sizesinttype)      end;    class function tarraydef.getreusable(def: tdef; elems: asizeint): tarraydef;      var        res: PHashSetItem;        oldsymtablestack: tsymtablestack;        arrdesc: packed record          def: tdef;          elecount: asizeint;        end;      begin        if not assigned(current_module) then          internalerror(2011081301);        arrdesc.def:=def;        arrdesc.elecount:=elems;        res:=current_module.arraydefs.FindOrAdd(@arrdesc,sizeof(arrdesc));        if not assigned(res^.Data) then          begin            { since these pointerdefs can be reused anywhere in the current              unit, add them to the global/staticsymtable (or local symtable              if they're a local def, because otherwise they'll be saved              to the ppu referencing a local symtable entry that doesn't              exist in the ppu) }            oldsymtablestack:=symtablestack;            { do not simply push/pop current_module.localsymtable, because              that can have side-effects (e.g., it removes helpers) }            symtablestack:=nil;            result:=carraydef.create(0,elems-1,sizesinttype);            result.elementdef:=def;            setup_reusable_def(def,result,res,oldsymtablestack);            { res^.Data may still be nil -> don't overwrite result }            exit;          end;        result:=tarraydef(res^.Data);      end;    class function tarraydef.getreusable_no_free(def: tdef; elems: asizeint): tarraydef;      begin        result:=getreusable(def,elems);        if not result.is_registered then          include(result.defoptions,df_not_registered_no_free);      end;    destructor tarraydef.destroy;      begin        symtable.free;        symtable:=nil;        inherited;      end;    constructor tarraydef.create_from_pointer(def:tpointerdef);      begin         { divide by the element size and do -1 so the array will have a valid size,           further, the element size might be 0 e.g. for empty records, so use max(...,1)           to avoid a division by zero }         self.create(0,(high(asizeint) div max(def.pointeddef.size,1))-1,           def.converted_pointer_to_array_range_type);         arrayoptions:=[ado_IsConvertedPointer];         setelementdef(def.pointeddef);      end;    constructor tarraydef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(arraydef,ppufile);         { the addresses are calculated later }         ppufile.getderef(_elementdefderef);         ppufile.getderef(rangedefderef);         lowrange:=ppufile.getasizeint;         highrange:=ppufile.getasizeint;         ppufile.getsmallset(arrayoptions);         ppuload_platform(ppufile);         symtable:=tarraysymtable.create(self);         tarraysymtable(symtable).ppuload(ppufile)      end;    function tarraydef.getcopy : tstoreddef;      begin        result:=carraydef.create(lowrange,highrange,rangedef);        tarraydef(result).arrayoptions:=arrayoptions;        tarraydef(result)._elementdef:=_elementdef;      end;    procedure tarraydef.buildderef;      begin        inherited buildderef;        tarraysymtable(symtable).buildderef;        _elementdefderef.build(_elementdef);        rangedefderef.build(rangedef);      end;    procedure tarraydef.deref;      begin        inherited deref;        tarraysymtable(symtable).deref(false);        _elementdef:=tdef(_elementdefderef.resolve);        rangedef:=tdef(rangedefderef.resolve);      end;    procedure tarraydef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         ppufile.putderef(_elementdefderef);         ppufile.putderef(rangedefderef);         ppufile.putasizeint(lowrange);         ppufile.putasizeint(highrange);         ppufile.putsmallset(arrayoptions);         writeentry(ppufile,ibarraydef);         tarraysymtable(symtable).ppuwrite(ppufile);      end;    function tarraydef.elesize : asizeint;      begin        if (ado_IsBitPacked in arrayoptions) then          internalerror(2006080101);        if assigned(_elementdef) then          result:=_elementdef.size        else          result:=0;      end;    function tarraydef.elepackedbitsize : asizeint;      begin        if not(ado_IsBitPacked in arrayoptions) then          internalerror(2006080102);	if assigned(_elementdef) then          result:=_elementdef.packedbitsize	else	  result:=0;      end;    function tarraydef.elecount : asizeuint;      var        qhigh,qlow : qword;      begin        if ado_IsDynamicArray in arrayoptions then          begin            result:=0;            exit;          end;        if (highrange>0) and (lowrange<0) then          begin            qhigh:=highrange;            qlow:=qword(-lowrange);            { prevent overflow, return 0 to indicate overflow }            if qhigh+qlow>qword(high(asizeint)-1) then              result:=0            else              result:=qhigh+qlow+1;          end        else          result:=int64(highrange)-lowrange+1;      end;    function tarraydef.size : asizeint;      var        cachedelecount : asizeuint;        cachedelesize : asizeint;      begin        if ado_IsDynamicArray in arrayoptions then          begin            size:=voidpointertype.size;            exit;          end;        { Tarraydef.size may never be called for an open array! }        if highrange<lowrange then          internalerror(99080501);        if not (ado_IsBitPacked in arrayoptions) then          cachedelesize:=elesize        else          cachedelesize := elepackedbitsize;        cachedelecount:=elecount;        if (cachedelesize = 0) then          begin            size := 0;            exit;          end;        if (cachedelecount = 0) then          begin            size := -1;            exit;          end;        { prevent overflow, return -1 to indicate overflow }        { also make sure we don't need 64/128 bit arithmetic to calculate offsets }        if (cachedelecount > asizeuint(high(asizeint))) or           ((high(asizeint) div cachedelesize) < asizeint(cachedelecount)) or           { also lowrange*elesize must be < high(asizeint) to prevent overflow when             accessing the array, see ncgmem (PFV) }           ((high(asizeint) div cachedelesize) < abs(lowrange)) then          begin            result:=-1;            exit;          end;        result:=cachedelesize*asizeint(cachedelecount);        if (ado_IsBitPacked in arrayoptions) then          { can't just add 7 and divide by 8, because that may overflow }          result:=result div 8 + ord((result mod 8)<>0);{$ifdef cpu16bitaddr}        if result>65535 then          begin            result:=-1;            exit;          end;{$endif cpu16bitaddr}      end;    procedure tarraydef.setelementdef(def:tdef);      begin        _elementdef:=def;        if not(               (ado_IsDynamicArray in arrayoptions) or               (ado_IsConvertedPointer in arrayoptions) or               (ado_IsConstructor in arrayoptions) or               (highrange<lowrange)	      ) and           (size=-1) then          Message(sym_e_segment_too_large);      end;    function tarraydef.alignment : shortint;      begin         { alignment of dyn. arrays doesn't depend on the element size }         if (ado_IsDynamicArray in arrayoptions) then           alignment:=voidpointertype.alignment         { alignment is the target alignment for the used load size }         else if (ado_IsBitPacked in arrayoptions) and            (elementdef.typ in [enumdef,orddef]) then           alignment:=cgsize_orddef(int_cgsize(packedbitsloadsize(elepackedbitsize))).alignment         { alignment is the alignment of the elements }         else           alignment:=elementdef.alignment      end;    function tarraydef.needs_inittable : boolean;      begin         needs_inittable:=(ado_IsDynamicArray in arrayoptions) or elementdef.needs_inittable;      end;    function tarraydef.needs_separate_initrtti : boolean;      begin        if ado_IsBitPacked in arrayoptions then          result:=false        else          result:=elementdef.needs_separate_initrtti;      end;    function tarraydef.GetTypeName : string;      begin         if (ado_IsConstString in arrayoptions) then           result:='Constant String'         else if (ado_isarrayofconst in arrayoptions) or                 (ado_isConstructor in arrayoptions) then           begin             if (ado_isvariant in arrayoptions) or ((highrange=-1) and (lowrange=0)) then               GetTypeName:='Array Of Const'             else               GetTypeName:='{Array Of Const/Constant Open} Array of '+elementdef.typename;           end         else if (ado_IsDynamicArray in arrayoptions) then           GetTypeName:='{Dynamic} Array Of '+elementdef.typename         else if ((highrange=-1) and (lowrange=0)) then           GetTypeName:='{Open} Array Of '+elementdef.typename         else           begin              result := '';              if (ado_IsBitPacked in arrayoptions) then                result:='BitPacked ';              if rangedef.typ=enumdef then                result:=result+'Array['+rangedef.typename+'] Of '+elementdef.typename              else                result:=result+'Array['+tostr(lowrange)+'..'+                  tostr(highrange)+'] Of '+elementdef.typename           end;      end;    function tarraydef.getmangledparaname : TSymStr;      begin         if ado_isarrayofconst in arrayoptions then          getmangledparaname:='array_of_const'         else          if ((highrange=-1) and (lowrange=0)) then           getmangledparaname:='array_of_'+elementdef.mangledparaname         else          internalerror(200204176);      end;    function tarraydef.is_publishable : boolean;      begin        Result:=ado_IsDynamicArray in arrayoptions;      end;{***************************************************************************                              tabstractrecorddef***************************************************************************}    constructor tabstractrecorddef.create(const n:string; dt:tdeftyp;doregister:boolean);      begin        inherited create(dt,doregister);        objname:=stringdup(upper(n));        objrealname:=stringdup(n);        objectoptions:=[];        if assigned(current_module.namespace) then          begin            import_lib:=stringdup(current_module.namespace^);            replace(import_lib^,'.','/');          end;      end;    constructor tabstractrecorddef.ppuload(dt:tdeftyp;ppufile:tcompilerppufile);      begin        inherited ppuload(dt,ppufile);        objrealname:=ppufile.getpshortstring;        objname:=stringdup(upper(objrealname^));        import_lib:=ppufile.getpshortstring;        { only used for external C++ classes and Java classes/records }        if (import_lib^='') then          stringdispose(import_lib);        ppufile.getsmallset(objectoptions);      end;    procedure tabstractrecorddef.ppuwrite(ppufile: tcompilerppufile);      begin        inherited ppuwrite(ppufile);        ppufile.putstring(objrealname^);        if assigned(import_lib) then          ppufile.putstring(import_lib^)        else          ppufile.putstring('');        ppufile.putsmallset(objectoptions);      end;    destructor tabstractrecorddef.destroy;      begin        stringdispose(objname);        stringdispose(objrealname);        stringdispose(import_lib);        tcinitcode.free;        inherited destroy;      end;    procedure tabstractrecorddef.buildderefimpl;      begin         inherited buildderefimpl;         if not (df_copied_def in defoptions) then           tstoredsymtable(symtable).buildderefimpl;      end;    procedure tabstractrecorddef.derefimpl;      begin        inherited derefimpl;        if not (df_copied_def in defoptions) then          tstoredsymtable(symtable).derefimpl(false);      end;    procedure tabstractrecorddef.check_forwards;      begin        { the defs of a copied def are defined for the original type only }        if not(df_copied_def in defoptions) then          tstoredsymtable(symtable).check_forwards;      end;    function tabstractrecorddef.find_procdef_bytype(pt:tproctypeoption): tprocdef;      var        i: longint;        sym: tsym;      begin        for i:=0 to symtable.SymList.Count-1 do          begin            sym:=tsym(symtable.SymList[i]);            if sym.typ=procsym then              begin                result:=tprocsym(sym).find_procdef_bytype(pt);                if assigned(result) then                  exit;              end;          end;          result:=nil;      end;    function tabstractrecorddef.GetSymtable(t:tGetSymtable):TSymtable;      begin        if t=gs_record then          GetSymtable:=symtable        else          GetSymtable:=nil;      end;    function tabstractrecorddef.is_packed:boolean;      begin        result:=tabstractrecordsymtable(symtable).is_packed;      end;    function tabstractrecorddef.RttiName: string;        function generate_full_paramname(maxlength:longint):string;          const            commacount : array[boolean] of longint = (0,1);          var            fullparas,            paramname : ansistring;            module : tmodule;            sym : ttypesym;            i : longint;          begin            { we want at least enough space for an ellipsis }            if maxlength<3 then              internalerror(2014121203);            fullparas:='';            for i:=0 to genericparas.count-1 do              begin                sym:=ttypesym(genericparas[i]);                module:=find_module_from_symtable(sym.owner);                if not assigned(module) then                  internalerror(2014121202);                paramname:=module.realmodulename^;                if sym.typedef.typ in [objectdef,recorddef] then                  paramname:=paramname+'.'+tabstractrecorddef(sym.typedef).rttiname                else                  paramname:=paramname+'.'+sym.typedef.typename;                if length(fullparas)+commacount[i>0]+length(paramname)>maxlength then                  begin                    if i>0 then                      fullparas:=fullparas+',...'                    else                      fullparas:=fullparas+'...';                    break;                  end;                { could we fit an ellipsis after this parameter if it should be too long? }                if (maxlength-(length(fullparas)+commacount[i>0]+length(paramname))<4) and (i<genericparas.count-1) then                  begin                    { then omit already this parameter }                    if i>0 then                      fullparas:=fullparas+',...'                    else                      fullparas:=fullparas+'...';                    break;                  end;                if i>0 then                  fullparas:=fullparas+',';                fullparas:=fullparas+paramname;              end;            result:=fullparas;          end;      var        nongeneric,        basename : string;        i,        remlength,        paramcount,        crcidx : longint;      begin        if rttistring='' then          begin            if is_specialization then              begin                rttistring:=OwnerHierarchyName;                { there should be two $ characters, one before the CRC and one before the count }                crcidx:=-1;                for i:=length(objrealname^) downto 1 do                  if objrealname^[i]='$' then                    begin                      crcidx:=i;                      break;                    end;                if crcidx<0 then                  internalerror(2014121201);                basename:=copy(objrealname^,1,crcidx-1);                split_generic_name(basename,nongeneric,paramcount);                rttistring:=rttistring+nongeneric+'<';                remlength:=255-length(rttistring)-1;                if remlength<4 then                  rttistring:=rttistring+'>'                else                  rttistring:=rttistring+generate_full_paramname(remlength)+'>';              end            else              if is_generic then                begin                  rttistring:=OwnerHierarchyName;                  split_generic_name(objrealname^,nongeneric,paramcount);                  rttistring:=rttistring+nongeneric+'<';                  { we don't want any ',' if there is only one parameter }                  for i:=0 to paramcount-0 do                    rttistring:=rttistring+',';                  rttistring:=rttistring+'>';                end              else                rttistring:=OwnerHierarchyName+objrealname^;          end;        result:=rttistring;      end;    function tabstractrecorddef.search_enumerator_get: tprocdef;      var        sym : tsym;        i : integer;        pd : tprocdef;        hashedid : THashedIDString;      begin        result:=nil;        hashedid.id:='GETENUMERATOR';        sym:=tsym(symtable.FindWithHash(hashedid));        if assigned(sym) and (sym.typ=procsym) then          begin            for i := 0 to Tprocsym(sym).ProcdefList.Count - 1 do            begin              pd := tprocdef(Tprocsym(sym).ProcdefList[i]);              if (pd.proctypeoption = potype_function) and                 (is_class_or_interface_or_object(pd.returndef) or is_record(pd.returndef)) and                 (pd.visibility >= vis_public) then              begin                result:=pd;                exit;              end;            end;          end;      end;    function tabstractrecorddef.search_enumerator_move: tprocdef;      var        sym : tsym;        i : integer;        pd : tprocdef;        hashedid : THashedIDString;      begin        result:=nil;        // first search for po_enumerator_movenext method modifier        // then search for public function MoveNext: Boolean        for i:=0 to symtable.SymList.Count-1 do          begin            sym:=TSym(symtable.SymList[i]);            if (sym.typ=procsym) then            begin              pd:=Tprocsym(sym).find_procdef_byoptions([po_enumerator_movenext]);              if assigned(pd) then                begin                  result:=pd;                  exit;                end;            end;          end;        hashedid.id:='MOVENEXT';        sym:=tsym(symtable.FindWithHash(hashedid));        if assigned(sym) and (sym.typ=procsym) then          begin            for i:=0 to Tprocsym(sym).ProcdefList.Count-1 do            begin              pd := tprocdef(Tprocsym(sym).ProcdefList[i]);              if (pd.proctypeoption = potype_function) and                 is_boolean(pd.returndef) and                 (pd.minparacount = 0) and                 (pd.visibility >= vis_public) then              begin                result:=pd;                exit;              end;            end;          end;      end;    function tabstractrecorddef.search_enumerator_current: tsym;      var        sym: tsym;        i: integer;        hashedid : THashedIDString;      begin        result:=nil;        // first search for ppo_enumerator_current property modifier        // then search for public property Current        for i:=0 to symtable.SymList.Count-1 do          begin            sym:=TSym(symtable.SymList[i]);            if (sym.typ=propertysym) and (ppo_enumerator_current in tpropertysym(sym).propoptions) then            begin              result:=sym;              exit;            end;          end;        hashedid.id:='CURRENT';        sym:=tsym(symtable.FindWithHash(hashedid));        if assigned(sym) and (sym.typ=propertysym) and           (sym.visibility >= vis_public) and not tpropertysym(sym).propaccesslist[palt_read].empty then          begin            result:=sym;            exit;          end;      end;    function tabstractrecorddef.jvm_full_typename(with_package_name: boolean): string;      var        st: tsymtable;        enclosingdef: tdef;      begin        if typ=objectdef then          result:=tobjectdef(self).objextname^        else if assigned(typesym) then          result:=typesym.realname        { have to generate anonymous nested type in current unit/class/record }        else          internalerror(2011032601);        { in case of specializations, add some extras to prevent name conflicts          with nested classes }        if df_specialization in defoptions then          result:='$'+result+'$specialization$';        st:=owner;        while assigned(st) and              (st.symtabletype in [objectsymtable,recordsymtable,localsymtable]) do          begin            { nested classes are named as "OuterClass$InnerClass" }            enclosingdef:=tdef(st.defowner);            if enclosingdef.typ=procdef then              result:=result+tprocdef(enclosingdef).procsym.realname+'$$'+tostr(tprocdef(enclosingdef).procsym.symid)+'$'            else if enclosingdef.typ=objectdef then              result:=tobjectdef(enclosingdef).objextname^+'$'+result            else if assigned(enclosingdef.typesym) then              result:=enclosingdef.typesym.realname+'$'+result            else              internalerror(2011060305);            st:=enclosingdef.owner;          end;        if with_package_name and           assigned(import_lib) then          result:=import_lib^+'/'+result;      end;    function tabstractrecorddef.contains_float_field: boolean;      var        i : longint;      begin        result:=true;        for i:=0 to symtable.symlist.count-1 do          begin            if (tsym(symtable.symlist[i]).typ<>fieldvarsym) or               (sp_static in tsym(symtable.symlist[i]).symoptions) then              continue;            if assigned(tfieldvarsym(symtable.symlist[i]).vardef) then              begin                if tstoreddef(tfieldvarsym(symtable.symlist[i]).vardef).is_fpuregable then                  exit;                { search recursively }                if (tstoreddef(tfieldvarsym(symtable.symlist[i]).vardef).typ=recorddef) and                  (tabstractrecorddef(tfieldvarsym(symtable.symlist[i]).vardef).contains_float_field) then                  exit;              end;          end;        result:=false;      end;    function tabstractrecorddef.contains_cross_aword_field: boolean;      var        i : longint;        foffset, fsize: aword;      begin        result:=true;        for i:=0 to symtable.symlist.count-1 do          begin            if (tsym(symtable.symlist[i]).typ<>fieldvarsym) or               (sp_static in tsym(symtable.symlist[i]).symoptions) then              continue;            if assigned(tfieldvarsym(symtable.symlist[i]).vardef) then              begin                if is_packed then                  begin                    foffset:=tfieldvarsym(symtable.symlist[i]).fieldoffset;                    fsize:=tfieldvarsym(symtable.symlist[i]).vardef.packedbitsize;                  end                else                  begin                    foffset:=tfieldvarsym(symtable.symlist[i]).fieldoffset*8;                    fsize:=tfieldvarsym(symtable.symlist[i]).vardef.size*8;                  end;                if (fsize>0) and ((foffset div (sizeof(aword)*8)) <> ((foffset+fsize-1) div (sizeof(aword)*8))) then                  exit;                { search recursively }                if (tstoreddef(tfieldvarsym(symtable.symlist[i]).vardef).typ=recorddef) and                  (tabstractrecorddef(tfieldvarsym(symtable.symlist[i]).vardef).contains_cross_aword_field) then                  exit;              end;          end;        result:=false;      end;{***************************************************************************                                  trecorddef***************************************************************************}    constructor trecorddef.create(const n:string; p:TSymtable);      begin         inherited create(n,recorddef,true);         symtable:=p;         { we can own the symtable only if nobody else owns a copy so far }         if symtable.refcount=1 then           symtable.defowner:=self;         isunion:=false;         cloneddefderef.reset;      end;    constructor trecorddef.create_global_internal(n: string; packrecords, recordalignmin, maxCrecordalign: shortint);      var        oldsymtablestack: tsymtablestack;        ts: ttypesym;        definedname: boolean;      begin        { construct name }        definedname:=n<>'';        if not definedname then          n:='$InternalRec'+tostr(current_module.deflist.count);        oldsymtablestack:=symtablestack;        { do not simply push/pop current_module.localsymtable, because          that can have side-effects (e.g., it removes helpers) }        symtablestack:=nil;        symtable:=trecordsymtable.create(n,packrecords,recordalignmin,maxCrecordalign);        symtable.defowner:=self;        isunion:=false;        inherited create(n,recorddef,true);        { if we specified a name, then we'll probably want to look up the          type again by name too -> create typesym }        ts:=nil;        if definedname then          begin            ts:=ctypesym.create(n,self,true);            { avoid hints about unused types (these may only be used for              typed constant data) }            ts.increfcount;          end;        if assigned(current_module.localsymtable) then          begin            current_module.localsymtable.insertdef(self);            if definedname then              current_module.localsymtable.insert(ts);          end        else          begin            current_module.globalsymtable.insertdef(self);            if definedname then              current_module.globalsymtable.insert(ts);          end;        symtablestack:=oldsymtablestack;        { don't create RTTI for internal types, these are not exported }        defstates:=defstates+[ds_rtti_table_written,ds_init_table_written];        include(defoptions,df_internal);      end;    function trecorddef.add_field_by_def(const optionalname: TIDString; def: tdef): tsym;      var        sym: tfieldvarsym;        name: TIDString;        pname: ^TIDString;      begin        if optionalname='' then          begin            name:='$f'+tostr(trecordsymtable(symtable).symlist.count);            pname:=@name          end        else          pname:=@optionalname;        sym:=cfieldvarsym.create(pname^,vs_value,def,[],true);        symtable.insert(sym);        trecordsymtable(symtable).addfield(sym,vis_hidden);        result:=sym;      end;    procedure trecorddef.add_fields_from_deflist(fieldtypes: tfplist);      var        i: longint;      begin        for i:=0 to fieldtypes.count-1 do          add_field_by_def('',tdef(fieldtypes[i]));      end;    constructor trecorddef.ppuload(ppufile:tcompilerppufile);      procedure readvariantrecdesc(var variantrecdesc : pvariantrecdesc);        var          i,j : longint;        begin         if ppufile.getbyte=1 then           begin             new(variantrecdesc);             ppufile.getderef(variantrecdesc^.variantselectorderef);             SetLength(variantrecdesc^.branches,ppufile.getasizeint);             for i:=0 to high(variantrecdesc^.branches) do               begin                 SetLength(variantrecdesc^.branches[i].values,ppufile.getasizeint);                 for j:=0 to high(variantrecdesc^.branches[i].values) do                   variantrecdesc^.branches[i].values[j]:=ppufile.getexprint;                 readvariantrecdesc(variantrecdesc^.branches[i].nestedvariant);               end;           end         else           variantrecdesc:=nil;        end;      begin         inherited ppuload(recorddef,ppufile);         if df_copied_def in defoptions then           begin             ppufile.getderef(cloneddefderef);             ppuload_platform(ppufile);           end         else           begin             symtable:=trecordsymtable.create(objrealname^,0,0,0);             trecordsymtable(symtable).fieldalignment:=shortint(ppufile.getbyte);             trecordsymtable(symtable).recordalignment:=shortint(ppufile.getbyte);             trecordsymtable(symtable).padalignment:=shortint(ppufile.getbyte);             trecordsymtable(symtable).usefieldalignment:=shortint(ppufile.getbyte);             trecordsymtable(symtable).recordalignmin:=shortint(ppufile.getbyte);             trecordsymtable(symtable).datasize:=ppufile.getasizeint;             trecordsymtable(symtable).paddingsize:=ppufile.getword;             ppufile.getsmallset(trecordsymtable(symtable).managementoperators);             { position of ppuload_platform call must correspond               to position of writeentry in ppuwrite method }             ppuload_platform(ppufile);             trecordsymtable(symtable).ppuload(ppufile);             { the variantrecdesc is needed only for iso-like new statements new(prec,1,2,3 ...);               but because iso mode supports no units, there is no need to store the variantrecdesc               in the ppu             }             // readvariantrecdesc(variantrecdesc);             { requires usefieldalignment to be set }             symtable.defowner:=self;           end;         isunion:=false;      end;    destructor trecorddef.destroy;      procedure free_variantrecdesc(var variantrecdesc : pvariantrecdesc);        var          i : longint;        begin         while assigned(variantrecdesc) do           begin             for i:=0 to high(variantrecdesc^.branches) do               begin                 free_variantrecdesc(variantrecdesc^.branches[i].nestedvariant);                 SetLength(variantrecdesc^.branches[i].values,0);               end;             SetLength(variantrecdesc^.branches,0);             dispose(variantrecdesc);             variantrecdesc:=nil;           end;        end;      begin         if assigned(variantrecdesc) then           free_variantrecdesc(variantrecdesc);         if assigned(symtable) then           begin             symtable.free;             symtable:=nil;           end;         inherited destroy;      end;    function trecorddef.getcopy : tstoreddef;      begin        result:=crecorddef.create(objrealname^,symtable.getcopy);        trecorddef(result).isunion:=isunion;        include(trecorddef(result).defoptions,df_copied_def);        if assigned(tcinitcode) then          trecorddef(result).tcinitcode:=tcinitcode.getcopy;         if assigned(import_lib) then           trecorddef(result).import_lib:=stringdup(import_lib^);      end;    function trecorddef.needs_inittable : boolean;      begin        { each record with managed field or with any management operator needs          init table }        needs_inittable:=(trecordsymtable(symtable).managementoperators<>[]) or          trecordsymtable(symtable).needs_init_final      end;    function trecorddef.needs_separate_initrtti : boolean;      begin        result:=true;      end;    function trecorddef.has_non_trivial_init_child(check_parent:boolean):boolean;      begin        result:=trecordsymtable(symtable).has_non_trivial_init;      end;    procedure trecorddef.buildderef;      begin         inherited buildderef;         if df_copied_def in defoptions then           cloneddefderef.build(symtable.defowner)         else           tstoredsymtable(symtable).buildderef;      end;    procedure trecorddef.deref;      begin         inherited deref;         { now dereference the definitions }         if df_copied_def in defoptions then           begin             cloneddef:=trecorddef(cloneddefderef.resolve);             symtable:=cloneddef.symtable.getcopy;           end         else           tstoredsymtable(symtable).deref(false);         { internal types, only load from the system unit }         if assigned(owner) and            assigned(owner.name) and            (owner.name^='SYSTEM') then           begin             { TGUID  }             if not assigned(rec_tguid) and                (upper(typename)='TGUID') then               rec_tguid:=self             { JMP_BUF }             else if not assigned(rec_jmp_buf) and                (upper(typename)='JMP_BUF') then               rec_jmp_buf:=self             else if not assigned(rec_exceptaddr) and                (upper(typename)='TEXCEPTADDR') then               rec_exceptaddr:=self;           end;      end;    procedure trecorddef.ppuwrite(ppufile:tcompilerppufile);      procedure writevariantrecdesc(variantrecdesc : pvariantrecdesc);        var          i,j : longint;        begin         if assigned(variantrecdesc) then           begin             ppufile.putbyte(1);             ppufile.putderef(variantrecdesc^.variantselectorderef);             ppufile.putasizeint(length(variantrecdesc^.branches));             for i:=0 to high(variantrecdesc^.branches) do               begin                 ppufile.putasizeint(length(variantrecdesc^.branches[i].values));                 for j:=0 to high(variantrecdesc^.branches[i].values) do                   ppufile.putexprint(variantrecdesc^.branches[i].values[j]);                 writevariantrecdesc(variantrecdesc^.branches[i].nestedvariant);               end;           end         else           ppufile.putbyte(0);        end;      begin         inherited ppuwrite(ppufile);         if df_copied_def in defoptions then           ppufile.putderef(cloneddefderef)         else           begin             ppufile.putbyte(byte(trecordsymtable(symtable).fieldalignment));             ppufile.putbyte(byte(trecordsymtable(symtable).recordalignment));             ppufile.putbyte(byte(trecordsymtable(symtable).padalignment));             ppufile.putbyte(byte(trecordsymtable(symtable).usefieldalignment));             ppufile.putbyte(byte(trecordsymtable(symtable).recordalignmin));             ppufile.putasizeint(trecordsymtable(symtable).datasize);             ppufile.putword(trecordsymtable(symtable).paddingsize);             ppufile.putsmallset(trecordsymtable(symtable).managementoperators);             { the variantrecdesc is needed only for iso-like new statements new(prec,1,2,3 ...);               but because iso mode supports no units, there is no need to store the variantrecdesc               in the ppu             }             // writevariantrecdesc(variantrecdesc);           end;         writeentry(ppufile,ibrecorddef);         if not(df_copied_def in defoptions) then           trecordsymtable(symtable).ppuwrite(ppufile);      end;    function trecorddef.size:asizeint;      begin        result:=trecordsymtable(symtable).datasize;      end;    function trecorddef.alignment:shortint;      begin        alignment:=trecordsymtable(symtable).recordalignment;      end;    function trecorddef.padalignment:shortint;      begin        padalignment := trecordsymtable(symtable).padalignment;      end;    function trecorddef.GetTypeName : string;      begin         GetTypeName:='<record type>'      end;{***************************************************************************                       TABSTRACTPROCDEF***************************************************************************}    constructor tabstractprocdef.create(dt:tdeftyp;level:byte;doregister:boolean);      begin         inherited create(dt,doregister);         parast:=tparasymtable.create(self,level);         paras:=nil;         minparacount:=0;         maxparacount:=0;         proctypeoption:=potype_none;         proccalloption:=pocall_none;         procoptions:=[];         returndef:=voidtype;         returndefderef.reset;         savesize:=sizeof(pint);         callerargareasize:=0;         calleeargareasize:=0;         has_paraloc_info:=callnoside;         funcretloc[callerside].init;         funcretloc[calleeside].init;         check_mark_as_nested;      end;    destructor tabstractprocdef.destroy;      begin         if assigned(paras) then           begin{$ifdef MEMDEBUG}             memprocpara.start;{$endif MEMDEBUG}             paras.free;             paras:=nil;{$ifdef MEMDEBUG}             memprocpara.stop;{$endif MEMDEBUG}          end;         if assigned(parast) then          begin{$ifdef MEMDEBUG}            memprocparast.start;{$endif MEMDEBUG}            parast.free;            parast:=nil;{$ifdef MEMDEBUG}            memprocparast.stop;{$endif MEMDEBUG}          end;         funcretloc[callerside].done;         funcretloc[calleeside].done;         inherited destroy;      end;    procedure tabstractprocdef.count_para(p:TObject;arg:pointer);      begin        if (tsym(p).typ<>paravarsym) then          exit;        inc(plongint(arg)^);        if not(vo_is_hidden_para in tparavarsym(p).varoptions) then         begin           if not assigned(tparavarsym(p).defaultconstsym) then             inc(minparacount);           inc(maxparacount);         end;      end;    procedure tabstractprocdef.insert_para(p:TObject;arg:pointer);      begin        if (tsym(p).typ<>paravarsym) then          exit;        paras.add(p);      end;    procedure tabstractprocdef.calcparas;      var        paracount : longint;      begin        { This can already be assigned when          we need to reresolve this unit (PFV) }        if assigned(paras) then          paras.free;        paras:=tparalist.create(false);        paracount:=0;        minparacount:=0;        maxparacount:=0;        parast.SymList.ForEachCall(@count_para,@paracount);        paras.capacity:=paracount;        { Insert parameters in table }        parast.SymList.ForEachCall(@insert_para,nil);        { Order parameters }        paras.sortparas;      end;    function tabstractprocdef.mangledprocparanames(oldlen : longint) : string;      var        crc  : dword;        hp   : TParavarsym;        hs   : TSymStr;        newlen,        i    : integer;      begin        result:='';        hp:=nil;        { add parameter types }        for i:=0 to paras.count-1 do         begin           hp:=tparavarsym(paras[i]);           if not(vo_is_hidden_para in hp.varoptions) then             result:=result+'$'+hp.vardef.mangledparaname;         end;        { add resultdef, add $$ as separator to make it unique from a          parameter separator }        if not is_void(returndef) then          result:=result+'$$'+returndef.mangledparaname;        newlen:=length(result)+oldlen;        { Replace with CRC if the parameter line is very long }        if (newlen-oldlen>12) and           ((newlen>100) or (newlen-oldlen>64)) then          begin            crc:=0;            for i:=0 to paras.count-1 do              begin                hp:=tparavarsym(paras[i]);                if not(vo_is_hidden_para in hp.varoptions) then                  begin                    hs:=hp.vardef.mangledparaname;                    crc:=UpdateCrc32(crc,hs[1],length(hs));                  end;              end;            if not is_void(returndef) then              begin                { add a little prefix so that x(integer; integer) is different from x(integer):integer }                hs:='$$'+returndef.mangledparaname;                crc:=UpdateCrc32(crc,hs[1],length(hs));              end;            result:='$crc'+hexstr(crc,8);          end;      end;    procedure tabstractprocdef.buildderef;      begin         { released procdef? }         if not assigned(parast) then           exit;         inherited buildderef;         returndefderef.build(returndef);         if po_explicitparaloc in procoptions then           funcretloc[callerside].buildderef;         { parast }         tparasymtable(parast).buildderef;      end;    procedure tabstractprocdef.deref;      begin         inherited deref;         returndef:=tdef(returndefderef.resolve);         if po_explicitparaloc in procoptions then           begin             funcretloc[callerside].deref;             has_paraloc_info:=callerside;            end         else           begin             { deref is called after loading from a ppu, but also after another               unit has been reloaded/recompiled and all references must be               re-resolved. Since the funcretloc contains a reference to a tdef,               reset it so that we won't try to access the stale def }             funcretloc[callerside].init;             has_paraloc_info:=callnoside;           end;         { parast }         tparasymtable(parast).deref(false);         { recalculated parameters }         calcparas;      end;    constructor tabstractprocdef.ppuload(dt:tdeftyp;ppufile:tcompilerppufile);      begin         inherited ppuload(dt,ppufile);         parast:=nil;         Paras:=nil;         minparacount:=0;         maxparacount:=0;         ppufile.getderef(returndefderef);         proctypeoption:=tproctypeoption(ppufile.getbyte);         proccalloption:=tproccalloption(ppufile.getbyte);         ppufile.getnormalset(procoptions);         funcretloc[callerside].init;         if po_explicitparaloc in procoptions then           funcretloc[callerside].ppuload(ppufile);         savesize:=sizeof(pint);         if (po_explicitparaloc in procoptions) then           has_paraloc_info:=callerside;      end;    procedure tabstractprocdef.ppuwrite(ppufile:tcompilerppufile);      var        oldintfcrc : boolean;      begin         { released procdef? }         if not assigned(parast) then           exit;         inherited ppuwrite(ppufile);         ppufile.putderef(returndefderef);         oldintfcrc:=ppufile.do_interface_crc;         ppufile.do_interface_crc:=false;         ppufile.putbyte(ord(proctypeoption));         ppufile.putbyte(ord(proccalloption));         ppufile.putnormalset(procoptions);         ppufile.do_interface_crc:=oldintfcrc;         if (po_explicitparaloc in procoptions) then           funcretloc[callerside].ppuwrite(ppufile);      end;    function tabstractprocdef.typename_paras(pno: tprocnameoptions) : ansistring;      var        hs,s  : ansistring;        hp    : TParavarsym;        hpc   : tconstsym;        first : boolean;        i,j   : integer;      begin        s:='';        first:=true;        for i:=0 to paras.count-1 do         begin           hp:=tparavarsym(paras[i]);           if not(vo_is_hidden_para in hp.varoptions) or              (pno_showhidden in pno) then            begin               if first then                begin                  s:=s+'(';                  first:=false;                end               else                s:=s+';';               if vo_is_hidden_para in hp.varoptions then                 s:=s+'<';               case hp.varspez of                 vs_var :                   s:=s+'var ';                 vs_const :                   s:=s+'const ';                 vs_out :                   s:=s+'out ';                 vs_constref :                   s:=s+'constref ';                 else                   ;               end;               if (pno_paranames in pno) then                 s:=s+hp.realname+':';               if hp.univpara then                 s:=s+'univ ';               if assigned(hp.vardef.typesym) then                 begin                   hs:=hp.vardef.typesym.realname;                   if hs[1]<>'$' then                     s:=s+hp.vardef.OwnerHierarchyName+hs                   else                     s:=s+hp.vardef.GetTypeName;                 end               else                 s:=s+hp.vardef.GetTypeName;               { default value }               if assigned(hp.defaultconstsym) then                begin                  hpc:=tconstsym(hp.defaultconstsym);                  hs:='';                  case hpc.consttyp of                    constwstring:                      begin                        if pcompilerwidestring(hpc.value.valueptr)^.len>0 then                          begin                            setlength(hs,pcompilerwidestring(hpc.value.valueptr)^.len);                            for j:=0 to pcompilerwidestring(hpc.value.valueptr)^.len-1 do                             begin                               if (ord(pcompilerwidestring(hpc.value.valueptr)^.data[j])<127) and                                  not(byte(pcompilerwidestring(hpc.value.valueptr)^.data[j]) in [0,10,13]) then                                 hs[j+1]:=char(pcompilerwidestring(hpc.value.valueptr)^.data[j])                               else                                 hs[j+1]:='.';                             end;                          end;                      end;                    conststring,                    constresourcestring :                      begin                        if hpc.value.len>0 then                          begin                            setLength(hs,hpc.value.len);                            { don't write past the end of hs if the constant                              is > 255 chars }                            move(hpc.value.valueptr^,hs[1],length(hs));                            { make sure that constant strings with newline chars                              don't create a linebreak in the assembler code,                              since comments are line-based. Also remove nulls                              because the comments are written as a pchar. }                            ReplaceCase(hs,#0,'.');                            ReplaceCase(hs,#10,'.');                            ReplaceCase(hs,#13,'.');                          end;                      end;                    constreal :                      str(pbestreal(hpc.value.valueptr)^,hs);                    constpointer :                      hs:=tostr(hpc.value.valueordptr);                    constord :                      begin                        if is_boolean(hpc.constdef) then                          begin                            if hpc.value.valueord<>0 then                             hs:='TRUE'                            else                             hs:='FALSE';                          end                        else                          hs:=tostr(hpc.value.valueord);                      end;                    constnil :                      hs:='nil';                    constset :                      hs:='<set>';                    constguid:                      hs:=guid2string(pguid(hpc.value.valueptr)^);                    constnone:                      internalerror(2019050704);                  end;                  if hs<>'' then                   s:=s+'=`'+hs+'`';                end;               if vo_is_hidden_para in hp.varoptions then                 s:=s+'>';             end;         end;        if not first then         s:=s+')';        if (po_varargs in procoptions) then         s:=s+';VarArgs';        typename_paras:=s;      end;    function tabstractprocdef.is_methodpointer:boolean;      begin        result:=false;      end;    function tabstractprocdef.is_addressonly:boolean;      begin        result:=true;      end;    function tabstractprocdef.no_self_node: boolean;      begin        Result:=([po_staticmethod,po_classmethod]<=procoptions)or                (proctypeoption in [potype_class_constructor,potype_class_destructor,potype_operator]);      end;    function tabstractprocdef.getcopyas(newtyp:tdeftyp;copytyp:tproccopytyp; const paraprefix: string): tstoreddef;      var        j, nestinglevel: longint;        pvs, npvs: tparavarsym;      begin        nestinglevel:=parast.symtablelevel;        if newtyp=procdef then          begin            if (copytyp<>pc_bareproc) then              result:=cprocdef.create(nestinglevel,true)            else              result:=cprocdef.create(normal_function_level,true);            tprocdef(result).visibility:=vis_public;          end        else          begin            result:=cprocvardef.create(nestinglevel);          end;        tabstractprocdef(result).returndef:=returndef;        tabstractprocdef(result).returndefderef:=returndefderef;        pvs:=nil;        npvs:=nil;        for j:=0 to parast.symlist.count-1 do          begin            case tsym(parast.symlist[j]).typ of              paravarsym:                begin                  pvs:=tparavarsym(parast.symlist[j]);                  { in case of bare proc, don't copy self, vmt or framepointer                    parameters }                  if (copytyp in [pc_bareproc,pc_normal_no_hidden]) and                     (([vo_is_self,vo_is_vmt,vo_is_parentfp,vo_is_result,vo_is_funcret]*pvs.varoptions)<>[]) then                    continue;                  if paraprefix='' then                    npvs:=cparavarsym.create(pvs.realname,pvs.paranr,pvs.varspez,                      pvs.vardef,pvs.varoptions)                  else if not(vo_is_high_para in pvs.varoptions) then                    npvs:=cparavarsym.create(paraprefix+pvs.realname,pvs.paranr,pvs.varspez,                      pvs.vardef,pvs.varoptions)                  else                    npvs:=cparavarsym.create('$high'+paraprefix+copy(pvs.name,5,length(pvs.name)),pvs.paranr,pvs.varspez,                      pvs.vardef,pvs.varoptions);                  npvs.defaultconstsym:=pvs.defaultconstsym;                  tabstractprocdef(result).parast.insert(npvs);                end;              constsym:                begin                  // ignore, reuse original constym. Should also be duplicated                  // be safe though                end;              symconst.typesym:                begin                  // reuse original, part of generic declaration                end              else                internalerror(201160604);              end;          end;        tabstractprocdef(result).savesize:=savesize;        if (typ<>procvardef) and           (newtyp=procvardef) then          begin            { procvars can't be (class)constructures/destructors etc }            if proctypeoption=potype_constructor then              begin                tabstractprocdef(result).returndef:=tdef(owner.defowner);                if not(is_implicit_pointer_object_type(returndef) or                   (returndef.typ<>objectdef)) then                  tabstractprocdef(result).returndef:=cpointerdef.getreusable(tabstractprocdef(result).returndef);                tabstractprocdef(result).proctypeoption:=potype_function;              end            else if is_void(returndef) then              tabstractprocdef(result).proctypeoption:=potype_procedure            else              tabstractprocdef(result).proctypeoption:=potype_function;          end        else          tabstractprocdef(result).proctypeoption:=proctypeoption;        tabstractprocdef(result).proccalloption:=proccalloption;        tabstractprocdef(result).procoptions:=procoptions;        if (copytyp=pc_bareproc) then          tabstractprocdef(result).procoptions:=tabstractprocdef(result).procoptions*[po_explicitparaloc,po_hascallingconvention,po_varargs,po_iocheck,po_has_importname,po_has_importdll];        if newtyp=procvardef then          tabstractprocdef(result).procoptions:=tabstractprocdef(result).procoptions-[po_has_importname,po_has_importdll];        if copytyp=pc_address_only then          include(tabstractprocdef(result).procoptions,po_addressonly);        tabstractprocdef(result).callerargareasize:=callerargareasize;        tabstractprocdef(result).calleeargareasize:=calleeargareasize;        tabstractprocdef(result).maxparacount:=maxparacount;        tabstractprocdef(result).minparacount:=minparacount;        if po_explicitparaloc in procoptions then          tabstractprocdef(result).funcretloc[callerside]:=funcretloc[callerside].getcopy;        { recalculate parameter info }        tabstractprocdef(result).has_paraloc_info:=callnoside;{$ifdef m68k}        tabstractprocdef(result).exp_funcretloc:=exp_funcretloc;{$endif}        if (typ=procdef) and           (newtyp=procvardef) and           (owner.symtabletype=ObjectSymtable) then          include(tprocvardef(result).procoptions,po_methodpointer);      end;    function tabstractprocdef.compatible_with_pointerdef_size(ptr: tpointerdef): boolean;      begin        result:=is_addressonly;      end;    procedure tabstractprocdef.check_mark_as_nested;      begin         { nested procvars require that nested functions use the Delphi-style           nested procedure calling convention }         if (parast.symtablelevel>normal_function_level) and            (m_nested_procvars in current_settings.modeswitches) then           include(procoptions,po_delphi_nested_cc);      end;    procedure tabstractprocdef.init_paraloc_info(side: tcallercallee);      begin        if (side in [callerside,callbothsides]) and           not(has_paraloc_info in [callerside,callbothsides]) then          begin            if not is_c_variadic(self) then              callerargareasize:=paramanager.create_paraloc_info(self,callerside)            else              callerargareasize:=paramanager.create_varargs_paraloc_info(self,callerside,nil);            if has_paraloc_info in [calleeside,callbothsides] then              has_paraloc_info:=callbothsides            else              has_paraloc_info:=callerside;          end;        if (side in [calleeside,callbothsides]) and           not(has_paraloc_info in [calleeside,callbothsides]) then          begin            if not is_c_variadic(self) then              calleeargareasize:=paramanager.create_paraloc_info(self,calleeside)            else              callerargareasize:=paramanager.create_varargs_paraloc_info(self,calleeside,nil);            if has_paraloc_info in [callerside,callbothsides] then              has_paraloc_info:=callbothsides            else              has_paraloc_info:=calleeside;          end;      end;    procedure tabstractprocdef.done_paraloc_info(side: tcallercallee);      var        i: longint;      begin        if (side in [callerside,callbothsides]) and           (has_paraloc_info in [callerside,callbothsides]) then          begin            funcretloc[callerside].done;            for i:=0 to paras.count-1 do              tparavarsym(paras[i]).paraloc[callerside].done;            if has_paraloc_info=callerside then              has_paraloc_info:=callnoside            else              has_paraloc_info:=calleeside;          end;        if (side in [calleeside,callbothsides]) and           (has_paraloc_info in [calleeside,callbothsides]) then          begin            funcretloc[calleeside].done;            for i:=0 to paras.count-1 do              tparavarsym(paras[i]).paraloc[calleeside].done;            if has_paraloc_info=calleeside then              has_paraloc_info:=callnoside            else              has_paraloc_info:=callerside;          end;      end;    function tabstractprocdef.stack_tainting_parameter(side: tcallercallee): boolean;      var        p: tparavarsym;        ploc: PCGParalocation;        i: longint;      begin        result:=false;        init_paraloc_info(side);        for i:=0 to parast.SymList.Count-1 do          if tsym(parast.SymList[i]).typ=paravarsym then            begin              p:=tparavarsym(parast.SymList[i]);              { check if no parameter is located on the stack }              if (is_open_array(p.vardef) or                 is_array_of_const(p.vardef)) and (p.varspez=vs_value) then                begin                  result:=true;                  exit;                end;              ploc:=p.paraloc[side].location;              while assigned(ploc) do                begin                  if (ploc^.loc=LOC_REFERENCE) then                    begin                      result:=true;                      exit                    end;                  ploc:=ploc^.next;                end;            end;      end;    function tabstractprocdef.is_pushleftright: boolean;      begin        result:=false;      end;    function tabstractprocdef.address_type: tdef;      begin        result:=voidcodepointertype;      end;    function tabstractprocdef.ofs_address_type:tdef;      begin        result:=address_type;      end;    procedure tabstractprocdef.declared_far;      begin        Message1(parser_w_proc_directive_ignored,'FAR');      end;    procedure tabstractprocdef.declared_near;      begin        Message1(parser_w_proc_directive_ignored,'NEAR');      end;{***************************************************************************                                  TPROCDEF***************************************************************************}    function tprocdef.GetResultName: PShortString;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010301);        result:=implprocdefinfo^.resultname;      end;    procedure tprocdef.SetResultName(AValue: PShortString);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010302);        implprocdefinfo^.resultname:=AValue;      end;    function tprocdef.GetParentFPInitBlock: tnode;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010303);        result:=implprocdefinfo^.parentfpinitblock;      end;    function tprocdef.GetParentFPStruct: tsym;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010304);        result:=implprocdefinfo^.parentfpstruct;      end;    function tprocdef.GetParentFPStructPtrType: tdef;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010305);        result:=implprocdefinfo^.parentfpstructptrtype;      end;    procedure tprocdef.SetParentFPInitBlock(AValue: tnode);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010306);        implprocdefinfo^.parentfpinitblock:=AValue;      end;    function tprocdef.Getprocendtai: tai;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010307);        result:=implprocdefinfo^.procendtai;      end;    function tprocdef.Getprocstarttai: tai;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010308);        result:=implprocdefinfo^.procstarttai;      end;    procedure tprocdef.Setprocendtai(AValue: tai);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010309);        implprocdefinfo^.procendtai:=AValue;      end;    function tprocdef.Getskpara: pointer;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010310);        result:=implprocdefinfo^.skpara;      end;    procedure tprocdef.Setskpara(AValue: pointer);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010311);        implprocdefinfo^.skpara:=AValue;      end;    function tprocdef.Getpersonality: tprocdef;      begin        if not assigned(implprocdefinfo) then          internalerror(2016121701);         result:=implprocdefinfo^.personality;      end;    procedure tprocdef.Setpersonality(AValue: tprocdef);      begin        if not assigned(implprocdefinfo) then          internalerror(2016121702);        implprocdefinfo^.personality:=AValue;      end;    function tprocdef.Getforwarddef: boolean;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010312);        result:=implprocdefinfo^.forwarddef;      end;    function tprocdef.Gethasforward: boolean;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010313);        result:=implprocdefinfo^.hasforward;      end;    function tprocdef.Getinterfacedef: boolean;      begin        if not assigned(implprocdefinfo) then          internalerror(2014010314);        result:=implprocdefinfo^.interfacedef;      end;    procedure tprocdef.Setforwarddef(AValue: boolean);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010315);        implprocdefinfo^.forwarddef:=AValue;      end;    procedure tprocdef.Sethasforward(AValue: boolean);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010316);        implprocdefinfo^.hasforward:=AValue;      end;    function tprocdef.GetIsEmpty: boolean;      begin        result:=pio_empty in implprocoptions;      end;    procedure tprocdef.SetIsEmpty(AValue: boolean);      begin        if AValue then          include(implprocoptions,pio_empty)        else          include(implprocoptions,pio_empty);      end;    function tprocdef.GetHasInliningInfo: boolean;      begin        result:=pio_has_inlininginfo in implprocoptions;      end;    procedure tprocdef.SetHasInliningInfo(AValue: boolean);      begin        if AValue then          include(implprocoptions,pio_has_inlininginfo)        else          exclude(implprocoptions,pio_has_inlininginfo);      end;    procedure tprocdef.Setinterfacedef(AValue: boolean);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010317);        implprocdefinfo^.interfacedef:=AValue;      end;    procedure tprocdef.Setprocstarttai(AValue: tai);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010318);        implprocdefinfo^.procstarttai:=AValue;      end;    procedure tprocdef.SetParentFPStruct(AValue: tsym);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010319);        implprocdefinfo^.parentfpstruct:=AValue;      end;    procedure tprocdef.SetParentFPStructPtrType(AValue: tdef);      begin        if not assigned(implprocdefinfo) then          internalerror(2014010320);        implprocdefinfo^.parentfpstructptrtype:=AValue;      end;    constructor tprocdef.create(level:byte;doregister:boolean);      begin         inherited create(procdef,level,doregister);         implprocdefinfo:=allocmem(sizeof(implprocdefinfo^));         localst:=tlocalsymtable.create(self,parast.symtablelevel);{$ifdef symansistr}         _mangledname:='';{$else symansistr}         _mangledname:=nil;{$endif symansistr}         fileinfo:=current_filepos;         extnumber:=$ffff;         aliasnames:=TCmdStrList.create;         funcretsym:=nil;         funcretsymderef.reset;         procsymderef.reset;         forwarddef:=true;         interfacedef:=false;         hasforward:=false;         struct := nil;         structderef.reset;         import_dll:=nil;         import_name:=nil;         import_nr:=0;         inlininginfo:=nil;         deprecatedmsg:=nil;      end;    constructor tprocdef.ppuload(ppufile:tcompilerppufile);      var        i,aliasnamescount,sizeleft : longint;        level : byte;        buf : array[0..255] of byte;      begin         inherited ppuload(procdef,ppufile);{$ifdef symansistr}         if po_has_mangledname in procoptions then           _mangledname:=ppufile.getansistring         else           _mangledname:='';{$else symansistr}         if po_has_mangledname in procoptions then          _mangledname:=ppufile.getpshortstring         else          _mangledname:=nil;{$endif symansistr}         extnumber:=ppufile.getword;         level:=ppufile.getbyte;         ppufile.getderef(structderef);         ppufile.getderef(procsymderef);         ppufile.getposinfo(fileinfo);         visibility:=tvisibility(ppufile.getbyte);         ppufile.getsmallset(symoptions);         if sp_has_deprecated_msg in symoptions then           deprecatedmsg:=ppufile.getpshortstring         else           deprecatedmsg:=nil;         { import stuff }         if po_has_importdll in procoptions then           import_dll:=ppufile.getpshortstring         else           import_dll:=nil;         if po_has_importname in procoptions then           import_name:=ppufile.getpshortstring         else           import_name:=nil;         import_nr:=ppufile.getword;         if (po_msgint in procoptions) then           messageinf.i:=ppufile.getlongint;         if (po_msgstr in procoptions) then           messageinf.str:=ppufile.getpshortstring;         if (po_dispid in procoptions) then           dispid:=ppufile.getlongint;         { inline stuff }         ppufile.getsmallset(implprocoptions);         if has_inlininginfo then           begin             ppufile.getderef(funcretsymderef);             new(inlininginfo);             ppufile.getsmallset(inlininginfo^.flags);           end         else           begin             inlininginfo:=nil;             funcretsym:=nil;           end;         aliasnames:=TCmdStrList.create;         { count alias names }         aliasnamescount:=ppufile.getbyte;         for i:=1 to aliasnamescount do           aliasnames.insert(ppufile.getstring);         { load the token stream containing the declaration }         sizeleft:=ppufile.getlongint;         if sizeleft>0 then           begin             init_genericdecl;             while sizeleft>0 do               begin                 if sizeleft>sizeof(buf) then                   i:=sizeof(buf)                 else                   i:=sizeleft;                 ppufile.getdata(buf,i);                 genericdecltokenbuf.write(buf,i);                 dec(sizeleft,i);               end;           end;         ppuload_platform(ppufile);         { load para symtable }         parast:=tparasymtable.create(self,level);         tparasymtable(parast).ppuload(ppufile);         { load local symtable }         if has_inlininginfo then          begin            localst:=tlocalsymtable.create(self,level);            tlocalsymtable(localst).ppuload(ppufile);          end         else          localst:=nil;         { inline stuff }         if has_inlininginfo then           inlininginfo^.code:=ppuloadnodetree(ppufile);         { default values for no persistent data }         if (cs_link_deffile in current_settings.globalswitches) and            (tf_need_export in target_info.flags) and            (po_exports in procoptions) then           deffile.AddExport(mangledname);         { Disable po_has_inlining until the derefimpl is done }         has_inlininginfo:=false;      end;    destructor tprocdef.destroy;      begin         aliasnames.free;         aliasnames:=nil;         if assigned(localst) and           (localst.symtabletype<>staticsymtable) then          begin{$ifdef MEMDEBUG}            memproclocalst.start;{$endif MEMDEBUG}            localst.free;            localst:=nil;{$ifdef MEMDEBUG}            memproclocalst.start;{$endif MEMDEBUG}          end;         if assigned(inlininginfo) then          begin{$ifdef MEMDEBUG}            memprocnodetree.start;{$endif MEMDEBUG}            tnode(inlininginfo^.code).free;{$ifdef MEMDEBUG}            memprocnodetree.start;{$endif MEMDEBUG}            dispose(inlininginfo);            inlininginfo:=nil;          end;         freeimplprocdefinfo;         genericdecltokenbuf.free;         genericdecltokenbuf:=nil;         stringdispose(import_dll);         stringdispose(import_name);         stringdispose(deprecatedmsg);         if (po_msgstr in procoptions) then           stringdispose(messageinf.str);{$ifndef symansistr}         if assigned(_mangledname) then          begin{$ifdef MEMDEBUG}            memmanglednames.start;{$endif MEMDEBUG}            stringdispose(_mangledname);{$ifdef MEMDEBUG}            memmanglednames.stop;{$endif MEMDEBUG}          end;{$endif symansistr}         inherited destroy;      end;    procedure tprocdef.freeimplprocdefinfo;      begin        if assigned(implprocdefinfo) then          begin            stringdispose(implprocdefinfo^.resultname);            freemem(implprocdefinfo);            implprocdefinfo:=nil;          end;      end;    procedure tprocdef.ppuwrite(ppufile:tcompilerppufile);      var        oldintfcrc : boolean;        aliasnamescount,i,sizeleft : longint;        item : TCmdStrListItem;        buf : array[0..255] of byte;      begin         { released procdef? }         if not assigned(parast) then           exit;         inherited ppuwrite(ppufile);{$ifdef symansistr}         if po_has_mangledname in procoptions then           ppufile.putansistring(_mangledname);{$else symansistr}         if po_has_mangledname in procoptions then          ppufile.putstring(_mangledname^);{$endif symansistr}         ppufile.putword(extnumber);         ppufile.putbyte(parast.symtablelevel);         ppufile.putderef(structderef);         ppufile.putderef(procsymderef);         ppufile.putposinfo(fileinfo);         ppufile.putbyte(byte(visibility));         ppufile.putsmallset(symoptions);         if sp_has_deprecated_msg in symoptions then           ppufile.putstring(deprecatedmsg^);         { import }         if po_has_importdll in procoptions then           ppufile.putstring(import_dll^);         if po_has_importname in procoptions then           ppufile.putstring(import_name^);         ppufile.putword(import_nr);         if (po_msgint in procoptions) then           ppufile.putlongint(messageinf.i);         if (po_msgstr in procoptions) then           ppufile.putstring(messageinf.str^);         if (po_dispid in procoptions) then           ppufile.putlongint(dispid);         { inline stuff }         oldintfcrc:=ppufile.do_crc;         ppufile.do_crc:=false;         ppufile.putsmallset(implprocoptions);         if has_inlininginfo then           begin             ppufile.putderef(funcretsymderef);             ppufile.putsmallset(inlininginfo^.flags);           end;         { count alias names }         aliasnamescount:=0;         item:=TCmdStrListItem(aliasnames.first);         while assigned(item) do           begin             inc(aliasnamescount);             item:=TCmdStrListItem(item.next);           end;         if aliasnamescount>255 then           internalerror(200711021);         ppufile.putbyte(aliasnamescount);         item:=TCmdStrListItem(aliasnames.first);         while assigned(item) do          begin            ppufile.putstring(item.str);            item:=TCmdStrListItem(item.next);          end;         ppufile.do_crc:=oldintfcrc;         { generic tokens for the declaration }         if assigned(genericdecltokenbuf) and (genericdecltokenbuf.size>0) then           begin             sizeleft:=genericdecltokenbuf.size;             genericdecltokenbuf.seek(0);             ppufile.putlongint(sizeleft);             while sizeleft>0 do               begin                 if sizeleft>sizeof(buf) then                   i:=sizeof(buf)                 else                   i:=sizeleft;                 genericdecltokenbuf.read(buf,i);                 ppufile.putdata(buf,i);                 dec(sizeleft,i);               end;           end         else           ppufile.putlongint(0);         { write this entry }         writeentry(ppufile,ibprocdef);         { Save the para symtable, this is taken from the interface }         tparasymtable(parast).ppuwrite(ppufile);         { save localsymtable for inline procedures or when local           browser info is requested, this has no influence on the crc }         if has_inlininginfo then          begin            oldintfcrc:=ppufile.do_crc;            ppufile.do_crc:=false;            tlocalsymtable(localst).ppuwrite(ppufile);            ppufile.do_crc:=oldintfcrc;          end;         { node tree for inlining }         oldintfcrc:=ppufile.do_crc;         ppufile.do_crc:=false;         if has_inlininginfo then           ppuwritenodetree(ppufile,inlininginfo^.code);         ppufile.do_crc:=oldintfcrc;      end;    function tprocdef.fullprocname(showhidden:boolean):string;      var        pno: tprocnameoptions;      begin        pno:=[];        if showhidden then          include(pno,pno_showhidden);        result:=customprocname(pno);      end;    function tprocdef.customprocname(pno: tprocnameoptions):ansistring;      var        s, rn : ansistring;        t : ttoken;        syssym : tsyssym;      begin{$ifdef EXTDEBUG}        include(pno,pno_showhidden);{$endif EXTDEBUG}        s:='';        if proctypeoption=potype_operator then          begin            for t:=NOTOKEN to last_overloaded do              if procsym.realname='$'+overloaded_names[t] then                begin                  s:='operator ';                  if (pno_ownername in pno) and                     assigned(struct) then                    s:=s+struct.RttiName+'.';                  s:=s+arraytokeninfo[t].str;                  if not (pno_noparams in pno) then                    s:=s+typename_paras(pno);                  break;                end;          end        else          begin            if (po_classmethod in procoptions) and               not(pno_noclassmarker in pno) then              s:='class ';            case proctypeoption of              potype_constructor,              potype_class_constructor:                s:=s+'constructor ';              potype_class_destructor,              potype_destructor:                s:=s+'destructor ';              else                if (pno_proctypeoption in pno) then                  begin                   if assigned(returndef) and                     not(is_void(returndef)) then                     s:=s+'function '                   else                     s:=s+'procedure ';                  end;            end;            if (pno_ownername in pno) and               (owner.symtabletype in [recordsymtable,objectsymtable]) then              s:=s+tabstractrecorddef(owner.defowner).RttiName+'.';            if (po_compilerproc in procoptions) and (extnumber<>$ffff) then              begin                syssym:=tsyssym.find_by_number(extnumber);                if not assigned(syssym) then                  internalerror(2016060305);                rn:=syssym.realname;              end            else              rn:=procsym.realname;            if (pno_noleadingdollar in pno) and               (rn[1]='$') then              delete(rn,1,1);            s:=s+rn;            if not (pno_noparams in pno) then              s:=s+typename_paras(pno);          end;        if not(proctypeoption in [potype_constructor,potype_destructor,             potype_class_constructor,potype_class_destructor]) and           assigned(returndef) and           not(is_void(returndef)) then          s:=s+':'+returndef.GetTypeName;        if assigned(owner) and (owner.symtabletype=localsymtable) then          s:=s+' is nested'        else if po_is_block in procoptions then          s:=s+' is block';        s:=s+';';        if po_far in procoptions then          s:=s+' far;';        { forced calling convention? }        if (po_hascallingconvention in procoptions) then          s:=s+' '+ProcCallOptionStr[proccalloption]+';';        if (po_staticmethod in procoptions) and           not (proctypeoption in [potype_class_constructor,potype_class_destructor]) then          s:=s+' Static;';        if pno_mangledname in pno then          s:=s+' -- mangled name: '+mangledname;        customprocname:=s;      end;    function tprocdef.is_methodpointer:boolean;      begin        { don't check assigned(_class), that's also the case for nested          procedures inside methods }        result:=(owner.symtabletype=ObjectSymtable)and not no_self_node;      end;    function tprocdef.is_addressonly:boolean;      begin        result:=assigned(owner) and                not is_methodpointer and                (not(m_nested_procvars in current_settings.modeswitches) or                 not is_nested_pd(self));      end;    procedure tprocdef.make_external;      begin        include(procoptions,po_external);        forwarddef:=false;      end;    procedure tprocdef.init_genericdecl;      begin        if assigned(genericdecltokenbuf) then          internalerror(2015061901);        genericdecltokenbuf:=tdynamicarray.create(256);      end;    function tprocdef.getfuncretsyminfo(out ressym: tsym; out resdef: tdef): boolean;      begin        result:=false;        if proctypeoption=potype_constructor then          begin            result:=true;            ressym:=tsym(parast.Find('self'));            resdef:=tabstractnormalvarsym(ressym).vardef;            { and TP-style constructors return a pointer to self }            if is_object(resdef) then              resdef:=cpointerdef.getreusable(resdef);          end        else if not is_void(returndef) then          begin            result:=true;            ressym:=funcretsym;            resdef:=tabstractnormalvarsym(ressym).vardef;          end;      end;    function tprocdef.has_alias_name(const s: TSymStr): boolean;      var        item : TCmdStrListItem;      begin        result:=true;        if mangledname=s then          exit;        item:=TCmdStrListItem(aliasnames.first);        while assigned(item) do          begin            if item.str=s then              exit;            item:=TCmdStrListItem(item.next);          end;        result:=false;      end;    function tprocdef.GetSymtable(t:tGetSymtable):TSymtable;      begin        case t of          gs_local :            GetSymtable:=localst;          gs_para :            GetSymtable:=parast;          else            GetSymtable:=nil;        end;      end;    function tprocdef.getcopyas(newtyp: tdeftyp; copytyp: tproccopytyp; const paraprefix: string): tstoreddef;      var        j : longint;      begin        result:=inherited;        if newtyp=procvardef then          begin            { create new paralist }            tprocvardef(result).calcparas;            exit;          end;        { don't copy mangled name, can be different }        tprocdef(result).messageinf:=messageinf;        tprocdef(result).dispid:=dispid;        if po_msgstr in procoptions then          tprocdef(result).messageinf.str:=stringdup(messageinf.str^);        tprocdef(result).symoptions:=symoptions;        if assigned(deprecatedmsg) then          tprocdef(result).deprecatedmsg:=stringdup(deprecatedmsg^);        { will have to be associated with appropriate procsym }        tprocdef(result).procsym:=nil;        { don't create aliases for bare copies, nor copy the funcretsym as          the function result parameter will be inserted again if necessary          (e.g. if the calling convention is changed) }        if not(copytyp in [pc_bareproc,pc_normal_no_hidden]) then          begin            tprocdef(result).aliasnames.concatListcopy(aliasnames);            if assigned(funcretsym) then              begin                if funcretsym.owner=parast then                  begin                    j:=parast.symlist.indexof(funcretsym);                    if j<0 then                      internalerror(2011040606);                    tprocdef(result).funcretsym:=tsym(tprocdef(result).parast.symlist[j]);                  end                else if funcretsym.owner=localst then                  begin                    { nothing to do, will be inserted for the new procdef while                      parsing its body (by pdecsub.insert_funcret_local) }                  end                else                  internalerror(2011040605);              end;          end;        { will have to be associated with a new struct }        tprocdef(result).struct:=nil;        if assigned(implprocdefinfo) then          begin            if assigned(resultname) then              tprocdef(result).resultname:=stringdup(resultname^);            tprocdef(result).synthetickind:=synthetickind;          end;        if assigned(import_dll) then          tprocdef(result).import_dll:=stringdup(import_dll^);        if assigned(import_name) then          tprocdef(result).import_name:=stringdup(import_name^);        tprocdef(result).import_nr:=import_nr;        tprocdef(result).extnumber:=$ffff;        tprocdef(result).visibility:=visibility;        { we need a separate implementation for the copied def }        tprocdef(result).forwarddef:=true;        tprocdef(result).interfacedef:=true;        { create new paralist }        tprocdef(result).calcparas;      end;    function tprocdef.getcopy: tstoreddef;      begin        result:=getcopyas(procdef,pc_normal,'');      end;    procedure tprocdef.buildderef;      begin         inherited buildderef;         structderef.build(struct);         { procsym that originaly defined this definition, should be in the           same symtable }         procsymderef.build(procsym);      end;    procedure tprocdef.buildderefimpl;      begin         inherited buildderefimpl;         { inline tree }         if has_inlininginfo then           begin             { Localst is not available for main/unit init }             if assigned(localst) then               begin                 tlocalsymtable(localst).buildderef;                 tlocalsymtable(localst).buildderefimpl;               end;             funcretsymderef.build(funcretsym);             inlininginfo^.code.buildderefimpl;           end;      end;    procedure tprocdef.deref;      begin         inherited deref;         struct:=tabstractrecorddef(structderef.resolve);         { procsym that originaly defined this definition, should be in the           same symtable }         procsym:=tprocsym(procsymderef.resolve);      end;    procedure tprocdef.derefimpl;      begin         { Enable has_inlininginfo when the inlininginfo           structure is available. The has_inlininginfo was disabled           after the load, since the data was invalid }         if assigned(inlininginfo) then           has_inlininginfo:=true;        { Inline }        if has_inlininginfo then          begin            { Locals }            if assigned(localst) then              begin                tlocalsymtable(localst).deref(false);                tlocalsymtable(localst).derefimpl(false);              end;            inlininginfo^.code.derefimpl;            { funcretsym, this is always located in the localst }            funcretsym:=tsym(funcretsymderef.resolve);          end        else          begin            { safety }            { Not safe! A unit may be reresolved after its interface has been              parsed but before its implementation has been parsed, and in that              case the funcretsym is still required!            funcretsym:=nil; }          end;      end;    function tprocdef.GetTypeName : string;      begin         GetTypeName := FullProcName(false);      end;    function tprocdef.mangledname : TSymStr;      begin{$ifdef symansistr}        if _mangledname='' then          begin            result:=defaultmangledname;            _mangledname:=result;          end        else          result:=_mangledname;{$else symansistr}        if not assigned(_mangledname) then          begin            result:=defaultmangledname;            _mangledname:=stringdup(mangledname);          end        else          result:=_mangledname^;{$endif symansistr}      end;    function tprocdef.defaultmangledname: TSymStr;      begin        { we need to use the symtable where the procsym is inserted,          because that is visible to the world }        defaultmangledname:=make_mangledname('',procsym.owner,procsym.name);        defaultmangledname:=defaultmangledname+mangledprocparanames(Length(defaultmangledname))      end;    function tprocdef.cplusplusmangledname : TSymStr;      function getcppparaname(p : tdef) : TSymStr;        const{$ifdef NAMEMANGLING_GCC2}           ordtype2str : array[tordtype] of string[2] = (             '',             'Uc','Us','Ui','Us','',             'Sc','s','i','x','',             'b','b','b','b','b','b',             'c','w','x');{$else NAMEMANGLING_GCC2}           ordtype2str : array[tordtype] of string[1] = (             'v',             'h','t','j','y','',             'a','s','i','x','',             'b','b','b','b','b',             'b','b','b','b',             'c','w','x','C');           floattype2str : array[tfloattype] of string[1] = (             'f','d','e','e',             'd','d','g');{$endif NAMEMANGLING_GCC2}        var           s : TSymStr;        begin           case p.typ of              orddef:                begin                  s:=ordtype2str[torddef(p).ordtype];                  if s='C' then                    s:=ordtype2str[range_to_basetype(torddef(p).low,torddef(p).high)];                end;              pointerdef:                s:='P'+getcppparaname(tpointerdef(p).pointeddef);{$ifndef NAMEMANGLING_GCC2}              floatdef:                s:=floattype2str[tfloatdef(p).floattype];{$endif NAMEMANGLING_GCC2}              else                internalerror(2103001);           end;           getcppparaname:=s;        end;      var         s,s2 : TSymStr;         hp   : TParavarsym;         i    : integer;      begin{$ifdef NAMEMANGLING_GCC2}        { outdated gcc 2.x name mangling scheme }         s := procsym.realname;         if procsym.owner.symtabletype=ObjectSymtable then           begin              s2:=upper(tobjectdef(procsym.owner.defowner).objrealname^);              case proctypeoption of                 potype_destructor:                   s:='_$_'+tostr(length(s2))+s2;                 potype_constructor:                   s:='___'+tostr(length(s2))+s2;                 else                   s:='_'+s+'__'+tostr(length(s2))+s2;              end;           end         else s:=s+'__';         s:=s+'F';         { concat modifiers }         { !!!!! }         { now we handle the parameters }         if maxparacount>0 then           begin             for i:=0 to paras.count-1 do               begin                 hp:=tparavarsym(paras[i]);                 { no hidden parameters form part of a C++ mangled name:                     a) self is not included                     b) there are no "high" or other hidden parameters                 }                 if vo_is_hidden_para in hp.varoptions then                   continue;                 s2:=getcppparaname(hp.vardef);                 if hp.varspez in [vs_var,vs_out] then                   s2:='R'+s2;                 s:=s+s2;               end;           end         else           s:=s+'v';         cplusplusmangledname:=s;{$else NAMEMANGLING_GCC2}         { gcc 3.x and 4.x name mangling scheme }         { see http://www.codesourcery.com/public/cxx-abi/abi.html#mangling }         if procsym.owner.symtabletype=ObjectSymtable then           begin             s:='_ZN';             s2:=tobjectdef(procsym.owner.defowner).objextname^;             s:=s+tostr(length(s2))+s2;             case proctypeoption of                potype_constructor:                  s:=s+'C1';                potype_destructor:                  s:=s+'D1';                else                  s:=s+tostr(length(procsym.realname))+procsym.realname;             end;             s:=s+'E';           end         else           s:=procsym.realname;         { now we handle the parameters }         if maxparacount>0 then           begin             for i:=0 to paras.count-1 do               begin                 hp:=tparavarsym(paras[i]);                 { no hidden parameters form part of a C++ mangled name:                     a) self is not included                     b) there are no "high" or other hidden parameters                 }                 if vo_is_hidden_para in hp.varoptions then                   continue;                 s2:=getcppparaname(hp.vardef);                 if hp.varspez in [vs_var,vs_out] then                   s2:='R'+s2;                 s:=s+s2;               end;           end         else           s:=s+'v';         cplusplusmangledname:=s;{$endif NAMEMANGLING_GCC2}      end;    function  tprocdef.objcmangledname : TSymStr;      var        manglednamelen: longint;        iscatmethod   : boolean;      begin        if not (po_msgstr in procoptions) then          internalerror(2009030901);        { we may very well need longer strings to handle these... }        manglednamelen:=length(tobjectdef(procsym.owner.defowner).objextname^)+          length('+"[ ]"')+length(messageinf.str^);        iscatmethod:=oo_is_classhelper in tobjectdef(procsym.owner.defowner).objectoptions;        if (iscatmethod) then          inc(manglednamelen,length(tobjectdef(procsym.owner.defowner).childof.objextname^)+length('()'));        if manglednamelen>255 then          Message1(parser_e_objc_message_name_too_long,messageinf.str^);        if not(po_classmethod in procoptions) then          result:='"-['        else          result:='"+[';        { quotes are necessary because the +/- otherwise confuse the assembler          into expecting a number        }        if iscatmethod then          result:=result+tobjectdef(procsym.owner.defowner).childof.objextname^+'(';        result:=result+tobjectdef(procsym.owner.defowner).objextname^;        if iscatmethod then          result:=result+')';        result:=result+' '+messageinf.str^+']"';      end;    procedure tprocdef.setmangledname(const s : TSymStr);      begin        { This is not allowed anymore, the forward declaration          already needs to create the correct mangledname, no changes          afterwards are allowed (PFV) }        { Exception: interface definitions in mode macpas, since in that }        {   case no reference to the old name can exist yet (JM)         }{$ifdef symansistr}        if _mangledname<>'' then          if ((m_mac in current_settings.modeswitches) and              (interfacedef)) then            _mangledname:=''          else            internalerror(200411171);{$else symansistr}        if assigned(_mangledname) then          if ((m_mac in current_settings.modeswitches) and              (interfacedef)) then            stringdispose(_mangledname)          else            internalerror(200411171);{$endif symansistr}{$ifdef jvm}        { this routine can be called for compilerproces. can't set mangled          name since it must be calculated, but it uses import_name when set          -> set that one }        import_name:=stringdup(s);        include(procoptions,po_has_importname);        include(procoptions,po_has_mangledname);{$else}  {$ifdef symansistr}        _mangledname:=s;  {$else symansistr}        _mangledname:=stringdup(s);  {$endif symansistr}{$endif jvm}        include(procoptions,po_has_mangledname);      end;    procedure tprocdef.setcompilerprocname;      begin        procsym.realname:='$'+lower(procsym.name);      end;{***************************************************************************                                 TPROCVARDEF***************************************************************************}    constructor tprocvardef.create(level:byte);      begin         inherited create(procvardef,level,true);      end;    class function tprocvardef.getreusableprocaddr(def: tabstractprocdef): tprocvardef;      var        res: PHashSetItem;        oldsymtablestack: tsymtablestack;      begin        if not assigned(current_module) then          internalerror(2011081301);        res:=current_module.procaddrdefs.FindOrAdd(@def,sizeof(def));        if not assigned(res^.Data) then          begin            { since these pointerdefs can be reused anywhere in the current              unit, add them to the global/staticsymtable (or local symtable              if they're a local def, because otherwise they'll be saved              to the ppu referencing a local symtable entry that doesn't              exist in the ppu) }            oldsymtablestack:=symtablestack;            { do not simply push/pop current_module.localsymtable, because              that can have side-effects (e.g., it removes helpers) }            symtablestack:=nil;            result:=tprocvardef(def.getcopyas(procvardef,pc_address_only,''));            setup_reusable_def(def,result,res,oldsymtablestack);            { res^.Data may still be nil -> don't overwrite result }            exit;          end;        result:=tprocvardef(res^.Data);      end;    class function tprocvardef.getreusableprocaddr_no_free(def: tabstractprocdef): tprocvardef;      begin        result:=getreusableprocaddr(def);        if not result.is_registered then          include(result.defoptions,df_not_registered_no_free);      end;    constructor tprocvardef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(procvardef,ppufile);         { load para symtable }         parast:=tparasymtable.create(self,ppufile.getbyte);         ppuload_platform(ppufile);         tparasymtable(parast).ppuload(ppufile);      end;    function tprocvardef.getcopy : tstoreddef;      var        i : tcallercallee;        j : longint;      begin        result:=cprocvardef.create(parast.symtablelevel);        tprocvardef(result).returndef:=returndef;        tprocvardef(result).returndefderef:=returndefderef;        tprocvardef(result).parast:=parast.getcopy;        tprocvardef(result).savesize:=savesize;        { create paralist copy }        calcparas;        tprocvardef(result).paras:=tparalist.create(false);        tprocvardef(result).paras.count:=paras.count;        for j:=0 to paras.count-1 do          tprocvardef(result).paras[j]:=paras[j];        tprocvardef(result).proctypeoption:=proctypeoption;        tprocvardef(result).proccalloption:=proccalloption;        tprocvardef(result).procoptions:=procoptions;        tprocvardef(result).callerargareasize:=callerargareasize;        tprocvardef(result).calleeargareasize:=calleeargareasize;        tprocvardef(result).maxparacount:=maxparacount;        tprocvardef(result).minparacount:=minparacount;        for i:=low(funcretloc) to high(funcretloc) do          tprocvardef(result).funcretloc[i]:=funcretloc[i].getcopy;        tprocvardef(result).has_paraloc_info:=has_paraloc_info;{$ifdef m68k}        tprocvardef(result).exp_funcretloc:=exp_funcretloc;{$endif}      end;    procedure tprocvardef.ppuwrite(ppufile:tcompilerppufile);      begin        inherited ppuwrite(ppufile);        { Save the para symtable level (necessary to distinguish nested          procvars) }        ppufile.putbyte(parast.symtablelevel);        { Write this entry }        writeentry(ppufile,ibprocvardef);        { Save the para symtable, this is taken from the interface }        tparasymtable(parast).ppuwrite(ppufile);      end;    function tprocvardef.GetSymtable(t:tGetSymtable):TSymtable;      begin        case t of          gs_para :            GetSymtable:=parast;          else            GetSymtable:=nil;        end;      end;    function tprocvardef.size : asizeint;      begin         { we return false for is_addressonly for a block (because it's not a           simple pointer to a function), but they are handled as implicit           pointers to a datastructure that contains everything ->           voidpointertype.size instead of voidcodepointertype.size }         if po_is_block in procoptions then           size:=voidpointertype.size         else if not is_addressonly then           begin             if is_nested_pd(self) then               size:=voidcodepointertype.size+parentfpvoidpointertype.size             else               size:=voidcodepointertype.size+voidpointertype.size;           end         else           size:=voidcodepointertype.size;      end;    function tprocvardef.is_methodpointer:boolean;      begin        result:=(po_methodpointer in procoptions);      end;    function tprocvardef.is_addressonly:boolean;      begin        result:=((not(po_methodpointer in procoptions) or (po_staticmethod in procoptions)) and                 not(po_is_block in procoptions) and                 not is_nested_pd(self)) or                (po_addressonly in procoptions);      end;    function tprocvardef.getmangledparaname:TSymStr;      begin        if not(po_methodpointer in procoptions) then          if not is_nested_pd(self) then            result:='procvar'          else            { we need the manglednames here, because nestedprocvars can be anonymous, e.g.              having not a type name or not an unique one, see webtbs/tw27515.pp              Further, use $_ ... _$ delimiters to avoid ambiguous names, see webtbs/tw27515.pp }            result:='$_nestedprovar'+mangledprocparanames(0)+'_$'        else          result:='procvarofobj'      end;    function tprocvardef.getcopyas(newtyp: tdeftyp; copytyp: tproccopytyp; const paraprefix: string): tstoreddef;      begin        result:=inherited;        tabstractprocdef(result).calcparas;      end;    function tprocvardef.is_publishable : boolean;      begin         is_publishable:=(po_methodpointer in procoptions);      end;    function tprocvardef.GetTypeName : string;      var        s: string;        pno : tprocnameoptions;      begin{$ifdef EXTDEBUG}         pno:=[pno_showhidden];{$else EXTDEBUG}         pno:=[];{$endif EXTDEBUG}         s:='<';         if po_is_block in procoptions then           s := s+'reference to'         else if po_classmethod in procoptions then           s := s+'class method type of'         else           if po_addressonly in procoptions then             s := s+'address of'           else             s := s+'procedure variable type of';         if assigned(returndef) and            (returndef<>voidtype) then           s:=s+' function'+typename_paras(pno)+':'+returndef.GetTypeName         else           s:=s+' procedure'+typename_paras(pno);         if po_methodpointer in procoptions then           s := s+' of object';         if is_nested_pd(self) then           s := s+' is nested';         if po_far in procoptions then           s := s+';far';         GetTypeName := s+';'+ProcCallOptionStr[proccalloption]+'>';      end;{***************************************************************************                              TOBJECTDEF***************************************************************************}   constructor tobjectdef.create(ot:tobjecttyp;const n:string;c:tobjectdef;doregister:boolean);     begin        inherited create(n,objectdef,doregister);        fcurrent_dispid:=0;        objecttype:=ot;        childof:=nil;        childofderef.reset;        vmt_fieldderef.reset;        extendeddefderef.reset;        cloneddefderef.reset;        if objecttype=odt_helper then          owner.includeoption(sto_has_helper);        symtable:=tObjectSymtable.create(self,n,current_settings.packrecords,          current_settings.alignment.recordalignmin,current_settings.alignment.maxCrecordalign);        { create space for vmt !! }        vmtentries:=TFPList.Create;        set_parent(c);        if objecttype in [odt_interfacecorba,odt_interfacecom,odt_dispinterface] then          prepareguid;        { setup implemented interfaces }        if objecttype in [odt_class,odt_objcclass,odt_objcprotocol,odt_javaclass,odt_interfacejava] then          ImplementedInterfaces:=TFPObjectList.Create(true)        else          ImplementedInterfaces:=nil;        writing_class_record_dbginfo:=false;     end;    constructor tobjectdef.ppuload(ppufile:tcompilerppufile);      var         i,         implintfcount : longint;         d, getterd : tderef;         ImplIntf : TImplementedInterface;         vmtentry : pvmtentry;      begin         inherited ppuload(objectdef,ppufile);         objecttype:=tobjecttyp(ppufile.getbyte);         helpertype:=thelpertype(ppufile.getbyte);         objextname:=ppufile.getpshortstring;         { only used for external Objective-C classes/protocols }         if (objextname^='') then           stringdispose(objextname);         symtable:=tObjectSymtable.create(self,objrealname^,0,0,0);         tObjectSymtable(symtable).datasize:=ppufile.getasizeint;         tObjectSymtable(symtable).paddingsize:=ppufile.getword;         tObjectSymtable(symtable).fieldalignment:=shortint(ppufile.getbyte);         tObjectSymtable(symtable).recordalignment:=shortint(ppufile.getbyte);         tObjectSymtable(symtable).recordalignmin:=shortint(ppufile.getbyte);         ppufile.getderef(vmt_fieldderef);         ppufile.getderef(childofderef);         { load guid }         iidstr:=nil;         if objecttype in [odt_interfacecom,odt_interfacecorba,odt_dispinterface] then           begin              new(iidguid);              ppufile.getguid(iidguid^);              iidstr:=ppufile.getpshortstring;           end;         abstractcnt:=ppufile.getlongint;         if objecttype=odt_helper then           ppufile.getderef(extendeddefderef);         vmtentries:=TFPList.Create;         vmtentries.count:=ppufile.getlongint;         for i:=0 to vmtentries.count-1 do           begin             ppufile.getderef(d);             new(vmtentry);             vmtentry^.procdef:=nil;             vmtentry^.procdefderef:=d;             vmtentry^.visibility:=tvisibility(ppufile.getbyte);             vmtentries[i]:=vmtentry;           end;         { load implemented interfaces }         if objecttype in [odt_class,odt_objcclass,odt_objcprotocol,odt_javaclass,odt_interfacejava] then           begin             ImplementedInterfaces:=TFPObjectList.Create(true);             implintfcount:=ppufile.getlongint;             for i:=0 to implintfcount-1 do               begin                 ppufile.getderef(d);                 ppufile.getderef(getterd);                 ImplIntf:=TImplementedInterface.Create_deref(d,getterd);                 ImplIntf.IOffset:=ppufile.getlongint;                 byte(ImplIntf.IType):=ppufile.getbyte;                 ImplementedInterfaces.Add(ImplIntf);               end;           end         else           ImplementedInterfaces:=nil;         if df_copied_def in defoptions then           begin             ppufile.getderef(cloneddefderef);             ppuload_platform(ppufile);           end         else           begin             ppuload_platform(ppufile);             tObjectSymtable(symtable).ppuload(ppufile);           end;         { handles the predefined class tobject  }         { the last TOBJECT which is loaded gets }         { it !                                  }         if (childof=nil) and            (objecttype in [odt_class,odt_javaclass]) and            (objname^='TOBJECT') then           class_tobject:=self;         if (childof=nil) and            (objecttype=odt_interfacecom) then            if (objname^='IUNKNOWN') then              interface_iunknown:=self            else            if (objname^='IDISPATCH') then              interface_idispatch:=self;         if (childof=nil) and            (objecttype=odt_objcclass) and            (objname^='PROTOCOL') then           objc_protocoltype:=self;         if (objecttype=odt_javaclass) and            not(oo_is_formal in objectoptions) then           begin             if (objname^='JLOBJECT') then               java_jlobject:=self             else if (objname^='JLTHROWABLE') then               java_jlthrowable:=self             else if (objname^='FPCBASERECORDTYPE') then               java_fpcbaserecordtype:=self             else if (objname^='JLSTRING') then               java_jlstring:=self             else if (objname^='ANSISTRINGCLASS') then               java_ansistring:=self             else if (objname^='SHORTSTRINGCLASS') then               java_shortstring:=self             else if (objname^='JLENUM') then               java_jlenum:=self             else if (objname^='JUENUMSET') then               java_juenumset:=self             else if (objname^='FPCBITSET') then               java_jubitset:=self             else if (objname^='FPCBASEPROCVARTYPE') then               java_procvarbase:=self;           end;         writing_class_record_dbginfo:=false;       end;    destructor tobjectdef.destroy;      begin         if assigned(symtable) then           begin             symtable.free;             symtable:=nil;           end;         stringdispose(objextname);         stringdispose(iidstr);         if assigned(ImplementedInterfaces) then           begin             ImplementedInterfaces.free;             ImplementedInterfaces:=nil;           end;         if assigned(iidguid) then           begin             dispose(iidguid);             iidguid:=nil;           end;         if assigned(vmtentries) then           begin             resetvmtentries;             vmtentries.free;             vmtentries:=nil;           end;         if assigned(vmcallstaticinfo) then           begin             freemem(vmcallstaticinfo);             vmcallstaticinfo:=nil;           end;         inherited destroy;      end;    function tobjectdef.getcopy : tstoreddef;      var        i : longint;      begin        result:=cobjectdef.create(objecttype,objrealname^,childof,true);        { the constructor allocates a symtable which we release to avoid memory leaks }        tobjectdef(result).symtable.free;        tobjectdef(result).symtable:=symtable.getcopy;        if assigned(objextname) then          tobjectdef(result).objextname:=stringdup(objextname^);        if assigned(import_lib) then          tobjectdef(result).import_lib:=stringdup(import_lib^);        tobjectdef(result).objectoptions:=objectoptions;        include(tobjectdef(result).defoptions,df_copied_def);        tobjectdef(result).extendeddef:=extendeddef;        if assigned(tcinitcode) then          tobjectdef(result).tcinitcode:=tcinitcode.getcopy;        tobjectdef(result).vmt_field:=vmt_field;        if assigned(iidguid) then          begin            new(tobjectdef(result).iidguid);            move(iidguid^,tobjectdef(result).iidguid^,sizeof(iidguid^));          end;        if assigned(iidstr) then          tobjectdef(result).iidstr:=stringdup(iidstr^);        tobjectdef(result).abstractcnt:=abstractcnt;        if assigned(ImplementedInterfaces) then          begin            for i:=0 to ImplementedInterfaces.count-1 do              tobjectdef(result).ImplementedInterfaces.Add(TImplementedInterface(ImplementedInterfaces[i]).Getcopy);          end;        if assigned(vmtentries) then          begin            tobjectdef(result).vmtentries:=TFPList.Create;            tobjectdef(result).copyvmtentries(self);          end;      end;    procedure tobjectdef.ppuwrite(ppufile:tcompilerppufile);      var         i : longint;         vmtentry : pvmtentry;         ImplIntf : TImplementedInterface;         old_do_indirect_crc: boolean;      begin         { if class1 in unit A changes, and class2 in unit B inherits from it           (so unit B uses unit A), then unit B with class2 will be recompiled.           However, if there is also a class3 in unit C that only depends on           unit B, then unit C will not be recompiled because nothing changed           to the interface of unit B. Nevertheless, unit C can indirectly           depend on unit A via derefs, and these must be updated -> the           indirect crc keeps track of such changes. }         old_do_indirect_crc:=ppufile.do_indirect_crc;         ppufile.do_indirect_crc:=true;         inherited ppuwrite(ppufile);         ppufile.putbyte(byte(objecttype));         ppufile.putbyte(byte(helpertype));         if assigned(objextname) then           ppufile.putstring(objextname^)         else           ppufile.putstring('');         ppufile.putasizeint(tObjectSymtable(symtable).datasize);         ppufile.putword(tObjectSymtable(symtable).paddingsize);         ppufile.putbyte(byte(tObjectSymtable(symtable).fieldalignment));         ppufile.putbyte(byte(tObjectSymtable(symtable).recordalignment));         ppufile.putbyte(byte(tObjectSymtable(symtable).recordalignmin));         ppufile.putderef(vmt_fieldderef);         ppufile.putderef(childofderef);         if objecttype in [odt_interfacecom,odt_interfacecorba,odt_dispinterface] then           begin              ppufile.putguid(iidguid^);              ppufile.putstring(iidstr^);           end;         ppufile.putlongint(abstractcnt);         if objecttype=odt_helper then           ppufile.putderef(extendeddefderef);         ppufile.putlongint(vmtentries.count);         for i:=0 to vmtentries.count-1 do           begin             vmtentry:=pvmtentry(vmtentries[i]);             ppufile.putderef(vmtentry^.procdefderef);             ppufile.putbyte(byte(vmtentry^.visibility));           end;         if assigned(ImplementedInterfaces) then           begin             ppufile.putlongint(ImplementedInterfaces.Count);             for i:=0 to ImplementedInterfaces.Count-1 do               begin                 ImplIntf:=TImplementedInterface(ImplementedInterfaces[i]);                 ppufile.putderef(ImplIntf.intfdefderef);                 ppufile.putderef(ImplIntf.ImplementsGetterDeref);                 ppufile.putlongint(ImplIntf.Ioffset);                 ppufile.putbyte(byte(ImplIntf.IType));               end;           end;         if df_copied_def in defoptions then           ppufile.putderef(cloneddefderef);         writeentry(ppufile,ibobjectdef);         if not(df_copied_def in defoptions) then           tObjectSymtable(symtable).ppuwrite(ppufile);         ppufile.do_indirect_crc:=old_do_indirect_crc;      end;    function tobjectdef.GetTypeName:string;      begin        { in this case we will go in endless recursion, because then  }        { there is no tsym associated yet with the def. It can occur  }        { (tests/webtbf/tw4757.pp), so for now give a generic name    }        { instead of the actual type name                             }        if not assigned(typesym) then          result:='<Currently Parsed Class>'        else          result:=typesymbolprettyname;      end;    procedure tobjectdef.buildderef;      var         i : longint;         vmtentry : pvmtentry;      begin         inherited buildderef;         vmt_fieldderef.build(vmt_field);         childofderef.build(childof);         if df_copied_def in defoptions then           cloneddefderef.build(symtable.defowner)         else           tstoredsymtable(symtable).buildderef;         if objecttype=odt_helper then           extendeddefderef.build(extendeddef);         for i:=0 to vmtentries.count-1 do           begin             vmtentry:=pvmtentry(vmtentries[i]);             vmtentry^.procdefderef.build(vmtentry^.procdef);           end;         if assigned(ImplementedInterfaces) then           begin             for i:=0 to ImplementedInterfaces.count-1 do               TImplementedInterface(ImplementedInterfaces[i]).buildderef;           end;      end;    procedure tobjectdef.deref;      var         i : longint;         vmtentry : pvmtentry;      begin         inherited deref;         vmt_field:=tsym(vmt_fieldderef.resolve);         childof:=tobjectdef(childofderef.resolve);         if df_copied_def in defoptions then           begin             cloneddef:=tobjectdef(cloneddefderef.resolve);             symtable:=cloneddef.symtable.getcopy;           end         else           tstoredsymtable(symtable).deref(false);         if objecttype=odt_helper then           extendeddef:=tdef(extendeddefderef.resolve);         for i:=0 to vmtentries.count-1 do           begin             vmtentry:=pvmtentry(vmtentries[i]);             vmtentry^.procdef:=tprocdef(vmtentry^.procdefderef.resolve);           end;         if assigned(ImplementedInterfaces) then           begin             for i:=0 to ImplementedInterfaces.count-1 do               TImplementedInterface(ImplementedInterfaces[i]).deref;           end;      end;    procedure create_class_helper_for_procdef(def: tobject; arg: pointer);      var        pd: tprocdef absolute def;        st: tsymtable;        psym: tsym;        nname: TIDString;      begin        if (tdef(def).typ<>procdef) then          exit;        { pd.owner = objcclass symtable -> defowner = objcclassdef ->          owner = symtable in which objcclassdef is defined        }        st:=pd.owner.defowner.owner;        nname:=class_helper_prefix+tprocsym(pd.procsym).name;        { check for an existing procsym with our special name }        psym:=tsym(st.find(nname));        if not assigned(psym) then          begin            psym:=cprocsym.create(nname);            { avoid warning about this symbol being unused }            psym.IncRefCount;            { don't check for duplicates:               a) we checked above               b) in case we are in the implementation section of a unit, this                  will also check for this symbol in the interface section                  (since you normally cannot have symbols with the same name                   both interface and implementation), and it's possible to                   have class helpers for the same class in the interface and                   in the implementation, and they cannot be merged since only                   the once in the interface must be saved to the ppu/visible                   from other units }            st.insert(psym,false);          end        else if (psym.typ<>procsym) then          internalerror(2009111501);        { add ourselves to this special procsym }        tprocsym(psym).procdeflist.add(def);      end;    procedure tobjectdef.derefimpl;      begin         inherited derefimpl;         { the procdefs are not owned by the class helper procsyms, so they           are not stored/restored either -> re-add them here }         if (objecttype in [odt_objcclass,odt_objcprotocol]) or            (oo_is_classhelper in objectoptions) then           symtable.DefList.ForEachCall(@create_class_helper_for_procdef,nil);      end;    procedure tobjectdef.resetvmtentries;      var        i : longint;      begin        for i:=0 to vmtentries.Count-1 do          Dispose(pvmtentry(vmtentries[i]));        vmtentries.clear;      end;    procedure tobjectdef.copyvmtentries(objdef:tobjectdef);      var        i : longint;        vmtentry : pvmtentry;      begin        resetvmtentries;        vmtentries.count:=objdef.vmtentries.count;        for i:=0 to objdef.vmtentries.count-1 do          begin            new(vmtentry);            vmtentry^:=pvmtentry(objdef.vmtentries[i])^;            vmtentries[i]:=vmtentry;          end;       end;    function tobjectdef.getparentdef:tdef;      begin{ TODO: Remove getparentdef hack}        { With 2 forward declared classes with the child class before the          parent class the child class is written earlier to the ppu. Leaving it          possible to have a reference to the parent class for property overriding,          but the parent class still has the childof not resolved yet (PFV) }        if childof=nil then          childof:=tobjectdef(childofderef.resolve);        result:=childof;      end;    procedure tobjectdef.prepareguid;      begin        { set up guid }        if not assigned(iidguid) then         begin            new(iidguid);            fillchar(iidguid^,sizeof(iidguid^),0); { default null guid }         end;        { setup iidstring }        if not assigned(iidstr) then          iidstr:=stringdup(''); { default is empty string }      end;    procedure tobjectdef.set_parent( c : tobjectdef);      begin        if assigned(childof) then          exit;        childof:=c;        if not assigned(c) then          exit;        { inherit options and status }        objectoptions:=objectoptions+(c.objectoptions*inherited_objectoptions);        { initially has the same number of abstract methods as the parent }        abstractcnt:=c.abstractcnt;        { add the data of the anchestor class/object }        if (objecttype in [odt_class,odt_object,odt_objcclass,odt_javaclass]) then          begin            tObjectSymtable(symtable).datasize:=tObjectSymtable(symtable).datasize+tObjectSymtable(c.symtable).datasize;            { inherit recordalignment }            tObjectSymtable(symtable).recordalignment:=tObjectSymtable(c.symtable).recordalignment;            { if both the parent and this record use C-alignment, also inherit              the current field alignment }            if (tObjectSymtable(c.symtable).usefieldalignment=C_alignment) and               (tObjectSymtable(symtable).usefieldalignment=C_alignment) then              tObjectSymtable(symtable).fieldalignment:=tObjectSymtable(c.symtable).fieldalignment;            { the padding is not inherited for Objective-C classes (maybe not              for cppclass either?) }            if objecttype=odt_objcclass then              tObjectSymtable(symtable).datasize:=tObjectSymtable(symtable).datasize-tObjectSymtable(c.symtable).paddingsize;            if (oo_has_vmt in objectoptions) and               (oo_has_vmt in c.objectoptions) then              tObjectSymtable(symtable).datasize:=tObjectSymtable(symtable).datasize-sizeof(pint);            { if parent has a vmt field then the offset is the same for the child PM }            if (oo_has_vmt in c.objectoptions) or is_class(self) then              begin                vmt_field:=c.vmt_field;                include(objectoptions,oo_has_vmt);              end;          end;      end;   procedure tobjectdef.insertvmt;     begin        if objecttype in [odt_interfacecom,odt_interfacecorba,odt_dispinterface,odt_objcclass,odt_objcprotocol,odt_javaclass,odt_interfacejava] then          exit;        if (oo_has_vmt in objectoptions) then          internalerror(12345)        else          begin             tObjectSymtable(symtable).datasize:=align(tObjectSymtable(symtable).datasize,                 tObjectSymtable(symtable).fieldalignment);             if (tf_requires_proper_alignment in target_info.flags) then               begin                 { Align VMT pointer and whole object instance if target CPU requires alignment. }                 tObjectSymtable(symtable).datasize:=align(tObjectSymtable(symtable).datasize,sizeof(pint));                 tObjectSymtable(symtable).alignrecord(tObjectSymtable(symtable).datasize,sizeof(pint));               end;             vmt_field:=cfieldvarsym.create('_vptr$'+objname^,vs_value,voidpointertype,[],true);             hidesym(vmt_field);             tObjectSymtable(symtable).insert(vmt_field);             tObjectSymtable(symtable).addfield(tfieldvarsym(vmt_field),vis_hidden);             include(objectoptions,oo_has_vmt);          end;     end;   function tobjectdef.vmt_offset: asizeint;     begin        if objecttype in [odt_interfacecom,odt_interfacecorba,odt_dispinterface,odt_objcclass,odt_objcprotocol,odt_javaclass,odt_interfacejava] then          result:=0        else if (tObjectSymtable(symtable).usefieldalignment<>bit_alignment) then          result:=tfieldvarsym(vmt_field).fieldoffset        else          result:=tfieldvarsym(vmt_field).fieldoffset div 8;     end;   procedure tobjectdef.check_forwards;     begin        if not(objecttype in [odt_interfacecom,odt_interfacecorba,odt_dispinterface,odt_objcprotocol,odt_interfacejava]) then          inherited;        if (oo_is_forward in objectoptions) then          begin             { ok, in future, the forward can be resolved }             Message1(sym_e_class_forward_not_resolved,objrealname^);             exclude(objectoptions,oo_is_forward);          end;     end;   function tobjectdef.find_destructor: tprocdef;     var       objdef: tobjectdef;     begin        objdef:=self;        while assigned(objdef) do          begin            result:=objdef.find_procdef_bytype(potype_destructor);            if assigned(result) then              exit;            objdef:=objdef.childof;          end;        result:=nil;     end;    function tobjectdef.implements_any_interfaces: boolean;      begin        result := (ImplementedInterfaces.Count > 0) or          (assigned(childof) and childof.implements_any_interfaces);      end;    function tobjectdef.size : asizeint;      begin        if objecttype in [odt_class,odt_interfacecom,odt_interfacecorba,odt_dispinterface,odt_objcclass,odt_objcprotocol,odt_helper,odt_javaclass,odt_interfacejava] then          result:=voidpointertype.size        else          result:=tObjectSymtable(symtable).datasize;      end;    function tobjectdef.alignment:shortint;      begin        if objecttype in [odt_class,odt_interfacecom,odt_interfacecorba,odt_dispinterface,odt_objcclass,odt_objcprotocol,odt_helper,odt_javaclass,odt_interfacejava] then          alignment:=voidpointertype.alignment        else          alignment:=tObjectSymtable(symtable).recordalignment;      end;    function tobjectdef.vmtmethodoffset(index:longint):longint;      begin        { for offset of methods for classes, see rtl/inc/objpash.inc }        case objecttype of        odt_class:          { the +2*sizeof(pint) is size and -size }          vmtmethodoffset:=index*voidcodepointertype.size+10*voidpointertype.size+2*sizeof(pint);        odt_helper,        odt_objcclass,        odt_objcprotocol:          vmtmethodoffset:=0;        odt_interfacecom,odt_interfacecorba,odt_dispinterface:          vmtmethodoffset:=index*voidcodepointertype.size;        odt_javaclass,        odt_interfacejava:          { invalid }          vmtmethodoffset:=-1;        else          { the +2*sizeof(pint) is size and -size }{$ifdef WITHDMT}          vmtmethodoffset:=index*voidcodepointertype.size+2*voidpointertype.size+2*sizeof(pint);{$else WITHDMT}          vmtmethodoffset:=index*voidcodepointertype.size+1*voidpointertype.size+2*sizeof(pint);{$endif WITHDMT}        end;      end;    function tobjectdef.vmt_mangledname : TSymStr;      begin        if not(oo_has_vmt in objectoptions) then          Message1(parser_n_object_has_no_vmt,objrealname^);        vmt_mangledname:=make_mangledname('VMT',owner,objname^);      end;    function tobjectdef.vmt_def: trecorddef;      var        vmttypesym: tsym;      begin        vmttypesym:=tsym(get_top_level_symtable.Find('vmtdef$'+mangledparaname));        if not assigned(vmttypesym) or           (vmttypesym.typ<>symconst.typesym) or           (ttypesym(vmttypesym).typedef.typ<>recorddef) then          internalerror(2015052501);        result:=trecorddef(ttypesym(vmttypesym).typedef);      end;    function tobjectdef.needs_inittable : boolean;      var        hp : tobjectdef;      begin         case objecttype of            odt_helper,            odt_class :              needs_inittable:=false;            odt_dispinterface,            odt_interfacecom:              needs_inittable:=true;            odt_interfacecorba:              begin                hp:=childof;                while assigned(hp) do                  begin                    if hp=interface_iunknown then                      begin                        needs_inittable:=true;                        exit;                      end;                    hp:=hp.childof;                  end;                needs_inittable:=false;              end;            odt_object:              needs_inittable:=                tObjectSymtable(symtable).needs_init_final or                (assigned(childof) and                 childof.needs_inittable);            odt_cppclass,            odt_objcclass,            odt_objcprotocol,            odt_javaclass,            odt_interfacejava:              needs_inittable:=false;            else              internalerror(200108267);         end;      end;    function tobjectdef.needs_separate_initrtti : boolean;      begin        result:=not (objecttype in [odt_interfacecom,odt_interfacecorba,odt_dispinterface]);      end;    function tobjectdef.has_non_trivial_init_child(check_parent:boolean):boolean;      begin        if objecttype in [odt_class,odt_object] then          begin            result:=tobjectsymtable(symtable).has_non_trivial_init or                      (check_parent and assigned(childof) and childof.has_non_trivial_init_child(true));          end        else          result:=false;      end;    function tobjectdef.rtti_mangledname(rt: trttitype): TSymStr;      begin        if not(objecttype in [odt_objcclass,odt_objcprotocol]) then          result:=inherited rtti_mangledname(rt)        else          begin            { necessary in case of a dynamic array of nsobject, or              if an nsobject field appears in a record that needs              init/finalisation }            if rt=initrtti then              begin                result:=voidpointertype.rtti_mangledname(rt);                exit;              end;            if not(target_info.system in systems_objc_nfabi) then              begin                result:=target_asm.labelprefix;                case objecttype of                  odt_objcclass:                    begin                      case rt of                        objcclassrtti:                          if not(oo_is_classhelper in objectoptions) then                            result:=result+'_OBJC_CLASS_'                          else                            result:=result+'_OBJC_CATEGORY_';                        objcmetartti:                          if not(oo_is_classhelper in objectoptions) then                            result:=result+'_OBJC_METACLASS_'                          else                            internalerror(2009111511);                        else                         internalerror(2009092302);                      end;                    end;                  odt_objcprotocol:                    result:=result+'_OBJC_PROTOCOL_';                  else                   ;                end;              end            else              begin                case objecttype of                  odt_objcclass:                    begin                      if (oo_is_classhelper in objectoptions) and                         (rt<>objcclassrtti) then                        internalerror(2009111512);                      case rt of                        objcclassrtti:                          if not(oo_is_classhelper in objectoptions) then                            result:='_OBJC_CLASS_$_'                          else                            result:='_OBJC_$_CATEGORY_';                        objcmetartti:                          result:='_OBJC_METACLASS_$_';                        objcclassrortti:                          result:=lower(target_asm.labelprefix)+'_OBJC_CLASS_RO_$_';                        objcmetarortti:                          result:=lower(target_asm.labelprefix)+'_OBJC_METACLASS_RO_$_';                        else                         internalerror(2009092303);                      end;                    end;                  odt_objcprotocol:                    begin                      result:=lower(target_asm.labelprefix);                      case rt of                        objcclassrtti:                          result:=result+'_OBJC_PROTOCOL_$_';                        objcmetartti:                          result:=result+'_OBJC_LABEL_PROTOCOL_$_';                        else                          internalerror(2009092501);                      end;                    end;                  else                    internalerror(2013113005);                end;              end;            result:=result+objextname^;          end;      end;    function tobjectdef.members_need_inittable : boolean;      begin        members_need_inittable:=tObjectSymtable(symtable).needs_init_final;      end;    function tobjectdef.is_publishable : boolean;      begin         is_publishable:=objecttype in [odt_class,odt_interfacecom,odt_interfacecorba,odt_dispinterface];      end;    function tobjectdef.get_next_dispid: longint;      begin        inc(fcurrent_dispid);        result:=fcurrent_dispid;      end;    function tobjectdef.search_enumerator_get: tprocdef;      begin        result:=inherited;        if not assigned(result) and assigned(childof) then          result:=childof.search_enumerator_get;      end;    function tobjectdef.search_enumerator_move: tprocdef;      begin        result:=inherited;        if not assigned(result) and assigned(childof) then          result:=childof.search_enumerator_move;      end;    function tobjectdef.search_enumerator_current: tsym;      begin        result:=inherited;        if not assigned(result) and assigned(childof) then          result:=childof.search_enumerator_current;      end;    procedure tobjectdef.register_created_classref_type;      begin        if not classref_created_in_current_module then          begin            classref_created_in_current_module:=true;            current_module.wpoinfo.addcreatedobjtypeforclassref(self);          end;      end;    procedure tobjectdef.register_created_object_type;      begin        if not created_in_current_module then          begin            created_in_current_module:=true;            current_module.wpoinfo.addcreatedobjtype(self);          end;      end;    procedure tobjectdef.register_maybe_created_object_type;      begin        { if we know it has been created for sure, no need          to also record that it maybe can be created in          this module        }        if not (created_in_current_module) and           not (maybe_created_in_current_module) then          begin            maybe_created_in_current_module:=true;            current_module.wpoinfo.addmaybecreatedbyclassref(self);          end;      end;    procedure tobjectdef.register_vmt_call(index: longint);      begin        if (is_object(self) or is_class(self)) then          current_module.wpoinfo.addcalledvmtentry(self,index);      end;    procedure check_and_finish_msg(data: tobject; arg: pointer);      var        def: tdef absolute data;        pd: tprocdef absolute data;        i,        paracount: longint;      begin        if (def.typ=procdef) then          begin            { add all messages also under a dummy name to the symtable in              which the objcclass/protocol/category is declared, so they can              be called via id.<name>            }            create_class_helper_for_procdef(pd,nil);            { we have to wait until now to set the mangled name because it              depends on the (possibly external) class name, which is defined              at the very end.  }            if not(po_msgstr in pd.procoptions) then              begin                CGMessagePos(pd.fileinfo,parser_e_objc_requires_msgstr);                { recover to avoid internalerror later on }                include(pd.procoptions,po_msgstr);                pd.messageinf.str:=stringdup('MissingDeclaration');              end;            { Mangled name is already set in case this is a copy of              another type.  }            if not(po_has_mangledname in pd.procoptions) then              begin                { check whether the number of formal parameters is correct,                  and whether they have valid Objective-C types }                paracount:=0;                for i:=1 to length(pd.messageinf.str^) do                  if pd.messageinf.str^[i]=':' then                    inc(paracount);                for i:=0 to pd.paras.count-1 do                  if not(vo_is_hidden_para in tparavarsym(pd.paras[i]).varoptions) and                     not is_array_of_const(tparavarsym(pd.paras[i]).vardef) then                    dec(paracount);                if (paracount<>0) then                  MessagePos(pd.fileinfo,sym_e_objc_para_mismatch);                pd.setmangledname(pd.objcmangledname);              end            else              { all checks already done }              exit;            if not(oo_is_external in pd.struct.objectoptions) then              begin                if (po_varargs in pd.procoptions) then                  MessagePos(pd.fileinfo,parser_e_varargs_need_cdecl_and_external)                else                  begin                    { check for "array of const" parameters }                    for i:=0 to pd.parast.symlist.count-1 do                      begin                        if (tsym(pd.parast.symlist[i]).typ=paravarsym) and                           is_array_of_const(tparavarsym(pd.parast.symlist[i]).vardef) then                          MessagePos(pd.fileinfo,parser_e_varargs_need_cdecl_and_external);                      end;                  end;              end;          end;      end;    procedure mark_private_fields_used(data: tobject; arg: pointer);      var        sym: tsym absolute data;      begin        if (sym.typ=fieldvarsym) and           (tfieldvarsym(sym).visibility in [vis_private,vis_strictprivate]) then          sym.IncRefCount;      end;    procedure tobjectdef.finish_objc_data;      begin        self.symtable.DefList.foreachcall(@check_and_finish_msg,nil);        if (oo_is_external in objectoptions) then          self.symtable.SymList.ForEachCall(@mark_private_fields_used,nil);      end;    procedure verify_objc_vardef(data: tobject; arg: pointer);      var        sym: tabstractvarsym absolute data;        res: pboolean absolute arg;        founderrordef: tdef;      begin        if not(tsym(data).typ in [paravarsym,fieldvarsym]) then          exit;        if (sym.typ=paravarsym) and           ((vo_is_hidden_para in tparavarsym(sym).varoptions) or            is_array_of_const(tparavarsym(sym).vardef)) then          exit;        if not objcchecktype(sym.vardef,founderrordef) then          begin            MessagePos1(sym.fileinfo,type_e_objc_type_unsupported,founderrordef.typename);            res^:=false;          end;      end;    procedure verify_objc_procdef_paras(data: tobject; arg: pointer);      var        def: tdef absolute data;        res: pboolean absolute arg;        founderrordef: tdef;      begin        if (def.typ<>procdef) then          exit;        { check parameter types for validity }        tprocdef(def).paras.foreachcall(@verify_objc_vardef,arg);        { check the result type for validity }        if not objcchecktype(tprocdef(def).returndef,founderrordef) then          begin            MessagePos1(tprocdef(def).funcretsym.fileinfo,type_e_objc_type_unsupported,founderrordef.typename);            res^:=false;          end;      end;    function tobjectdef.check_objc_types: boolean;      begin        { done in separate step from finish_objc_data, because when          finish_objc_data is called, not all forwarddefs have been resolved          yet and we need to know all types here }        result:=true;        self.symtable.symlist.foreachcall(@verify_objc_vardef,@result);        self.symtable.deflist.foreachcall(@verify_objc_procdef_paras,@result);      end;    procedure do_cpp_import_info(data: tobject; arg: pointer);      var        def: tdef absolute data;        pd: tprocdef absolute data;      begin        if (def.typ=procdef) then          begin            pd.setmangledname(target_info.Cprefix+pd.cplusplusmangledname);            if (oo_is_external in pd.struct.objectoptions) then              begin                { copied from psub.read_proc }                if assigned(tobjectdef(pd.struct).import_lib) then                   current_module.AddExternalImport(tobjectdef(pd.struct).import_lib^,pd.mangledname,pd.mangledname,0,false,false)                 else                   begin                     { add import name to external list for DLL scanning }                     if tf_has_dllscanner in target_info.flags then                       current_module.dllscannerinputlist.Add(pd.mangledname,pd);                   end;              end;          end;      end;    procedure tobjectdef.finish_cpp_data;      begin        self.symtable.DefList.ForEachCall(@do_cpp_import_info,nil);      end;{****************************************************************************                             TImplementedInterface****************************************************************************}    function TImplementedInterface.GetIOffset: longint;      begin        if (fIOffset=-1) and           (IType in [etFieldValue,etFieldValueClass]) then          result:=tfieldvarsym(ImplementsField).fieldoffset        else          result:=fIOffset;      end;    constructor TImplementedInterface.create(aintf: tobjectdef);      begin        inherited create;        intfdef:=aintf;        intfdefderef.reset;        IOffset:=-1;        IType:=etStandard;        NameMappings:=nil;        procdefs:=nil;      end;    constructor TImplementedInterface.create_deref(intfd,getterd:tderef);      begin        inherited create;        intfdef:=nil;        intfdefderef:=intfd;        ImplementsGetterDeref:=getterd;        IOffset:=-1;        IType:=etStandard;        NameMappings:=nil;        procdefs:=nil;      end;    destructor  TImplementedInterface.destroy;      var        i : longint;        mappedname : pshortstring;      begin        if assigned(NameMappings) then          begin            for i:=0 to NameMappings.Count-1 do              begin                mappedname:=pshortstring(NameMappings[i]);                stringdispose(mappedname);              end;            NameMappings.free;            NameMappings:=nil;          end;        if assigned(procdefs) then          begin            procdefs.free;            procdefs:=nil;          end;        inherited destroy;      end;    procedure TImplementedInterface.buildderef;      begin        intfdefderef.build(intfdef);        ImplementsGetterDeref.build(ImplementsGetter);      end;    procedure TImplementedInterface.deref;      begin        intfdef:=tobjectdef(intfdefderef.resolve);        ImplementsGetter:=tsym(ImplementsGetterDeref.resolve);      end;    procedure TImplementedInterface.AddMapping(const origname,newname: string);      begin        if not assigned(NameMappings) then          NameMappings:=TFPHashList.Create;        NameMappings.Add(origname,stringdup(newname));      end;    function TImplementedInterface.GetMapping(const origname: string):string;      var        mappedname : pshortstring;      begin        result:='';        if not assigned(NameMappings) then          exit;        mappedname:=PShortstring(NameMappings.Find(origname));        if assigned(mappedname) then          result:=mappedname^;      end;    procedure TImplementedInterface.AddImplProc(pd:tprocdef);      begin        if not assigned(procdefs) then          procdefs:=TFPObjectList.Create(false);        { duplicate entries must be stored, because multiple }        { interfaces can declare methods with the same name  }        { and all of these get their own VMT entry           }        procdefs.Add(pd);      end;    function TImplementedInterface.IsImplMergePossible(MergingIntf:TImplementedInterface;out weight: longint): boolean;      var        i : longint;      begin        result:=false;        { interfaces being implemented through delegation are not mergable (FK) }        if (IType<>etStandard) or (MergingIntf.IType<>etStandard) or not(assigned(ProcDefs)) or not(assigned(MergingIntf.ProcDefs)) then          exit;        weight:=0;        { empty interface is mergeable }        if ProcDefs.Count=0 then          begin            result:=true;            exit;          end;        { The interface to merge must at least the number of          procedures of this interface }        if MergingIntf.ProcDefs.Count<ProcDefs.Count then          exit;        for i:=0 to ProcDefs.Count-1 do          begin            if MergingIntf.ProcDefs[i]<>ProcDefs[i] then              exit;          end;        weight:=ProcDefs.Count;        result:=true;      end;    function TImplementedInterface.getcopy:TImplementedInterface;      begin        Result:=TImplementedInterface.Create(nil);        { 1) the procdefs list will be freed once for each copy          2) since the procdefs list owns its elements, those will also be freed for each copy          3) idem for the name mappings        }        { warning: this is completely wrong on so many levels...        Move(pointer(self)^,pointer(result)^,InstanceSize);        We need to make clean copies of the different fields        this is not implemented yet, and thus we generate an internal        error instead PM 2011-06-14 }        internalerror(2011061401);      end;{****************************************************************************                                TFORWARDDEF****************************************************************************}   constructor tforwarddef.create(const s:string;const pos:tfileposinfo);     begin        inherited create(forwarddef,true);        tosymname:=stringdup(s);        forwardpos:=pos;     end;    function tforwarddef.GetTypeName:string;      begin        GetTypeName:='unresolved forward to '+tosymname^;      end;    destructor tforwarddef.destroy;      begin        stringdispose(tosymname);        inherited destroy;      end;    function tforwarddef.getcopy:tstoreddef;      begin        result:=cforwarddef.create(tosymname^, forwardpos);      end;{****************************************************************************                               TUNDEFINEDDEF****************************************************************************}   constructor tundefineddef.create(doregister:boolean);     begin        inherited create(undefineddef,doregister);     end;    constructor tundefineddef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(undefineddef,ppufile);         ppuload_platform(ppufile);      end;    function tundefineddef.GetTypeName:string;      begin        GetTypeName:='<undefined type>';      end;    procedure tundefineddef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwrite(ppufile);         writeentry(ppufile,ibundefineddef);      end;{****************************************************************************                                  TERRORDEF****************************************************************************}    constructor terrordef.create;      begin        inherited create(errordef,true);        { prevent consecutive faults }        savesize:=1;      end;    procedure terrordef.ppuwrite(ppufile:tcompilerppufile);      begin        { Can't write errordefs to ppu }        internalerror(200411063);      end;    function terrordef.GetTypeName:string;      begin        GetTypeName:='<erroneous type>';      end;    function terrordef.getmangledparaname:TSymStr;      begin        getmangledparaname:='error';      end;{****************************************************************************                           Definition Helpers****************************************************************************}    function is_interfacecom(def: tdef): boolean;      begin        is_interfacecom:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_interfacecom);      end;    function is_interfacecom_or_dispinterface(def: tdef): boolean;      begin        is_interfacecom_or_dispinterface:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_interfacecom,odt_dispinterface]);      end;    function is_any_interface_kind(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          ((tobjectdef(def).objecttype in [odt_interfacecom,odt_dispinterface,odt_interfacecorba,odt_interfacejava,odt_objcprotocol]) or           is_objccategory(def));      end;    function is_interfacecorba(def: tdef): boolean;      begin        is_interfacecorba:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_interfacecorba);      end;    function is_interface(def: tdef): boolean;      begin        is_interface:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_interfacecom,odt_interfacecorba]);      end;    function is_dispinterface(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_dispinterface);      end;    function is_class(def: tdef): boolean;      begin        is_class:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_class);      end;    function is_object(def: tdef): boolean;      begin        is_object:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_object);      end;    function is_cppclass(def: tdef): boolean;      begin        is_cppclass:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_cppclass);      end;    function is_objcclass(def: tdef): boolean;      begin        is_objcclass:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_objcclass);      end;    function is_objectpascal_helper(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_helper);      end;    function is_objcclassref(def: tdef): boolean;      begin        is_objcclassref:=          assigned(def) and          (def.typ=classrefdef) and          is_objcclass(tclassrefdef(def).pointeddef);      end;    function is_objcprotocol(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_objcprotocol);      end;    function is_objccategory(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          { if used as a forward type }          ((tobjectdef(def).objecttype=odt_objccategory) or          { if used as after it has been resolved }           ((tobjectdef(def).objecttype=odt_objcclass) and            (oo_is_classhelper in tobjectdef(def).objectoptions)));      end;    function is_objc_class_or_protocol(def: tdef): boolean;      begin         result:=           assigned(def) and           (def.typ=objectdef) and           (tobjectdef(def).objecttype in [odt_objcclass,odt_objcprotocol]);      end;    function is_objc_protocol_or_category(def: tdef): boolean;      begin         result:=           assigned(def) and           (def.typ=objectdef) and           ((tobjectdef(def).objecttype = odt_objcprotocol) or            ((tobjectdef(def).objecttype = odt_objcclass) and             (oo_is_classhelper in tobjectdef(def).objectoptions)));      end;    function is_classhelper(def: tdef): boolean;      begin         result:=           is_objectpascal_helper(def) or           is_objccategory(def);      end;    function is_class_or_interface(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_class,odt_interfacecom,odt_interfacecorba]);      end;    function is_class_or_interface_or_objc(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_class,odt_interfacecom,odt_interfacecorba,odt_objcclass,odt_objcprotocol]);      end;    function is_class_or_interface_or_objc_or_java(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_class,odt_interfacecom,odt_interfacecorba,odt_objcclass,odt_objcprotocol,odt_javaclass,odt_interfacejava]);      end;    function is_class_or_interface_or_dispinterface_or_objc_or_java(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_class,odt_interfacecom,odt_interfacecorba,odt_dispinterface,odt_objcclass,odt_objcprotocol,odt_javaclass,odt_interfacejava]);      end;    function is_class_or_interface_or_object(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_class,odt_interfacecom,odt_interfacecorba,odt_object]);      end;    function is_class_or_interface_or_dispinterface(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_class,odt_interfacecom,odt_interfacecorba,odt_dispinterface]);      end;    function is_implicit_pointer_object_type(def: tdef): boolean;      begin        result:=          assigned(def) and          (((def.typ=objectdef) and            (tobjectdef(def).objecttype in [odt_class,odt_interfacecom,odt_interfacecorba,odt_dispinterface,odt_objcclass,odt_objcprotocol,odt_helper,odt_javaclass,odt_interfacejava])) or           ((target_info.system in systems_jvm) and            (def.typ=recorddef)));      end;    function is_implicit_array_pointer(def: tdef): boolean;      begin        result:=is_dynamic_array(def) or is_dynamicstring(def);      end;    function is_class_or_object(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_class,odt_object]);      end;    function is_record(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=recorddef);      end;    function is_javaclass(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_javaclass);      end;    function is_javaclassref(def: tdef): boolean;      begin        is_javaclassref:=          assigned(def) and          (def.typ=classrefdef) and          is_javaclass(tclassrefdef(def).pointeddef);      end;    function is_javainterface(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype=odt_interfacejava);      end;    function is_java_class_or_interface(def: tdef): boolean;      begin        result:=          assigned(def) and          (def.typ=objectdef) and          (tobjectdef(def).objecttype in [odt_javaclass,odt_interfacejava]);      end;    procedure loadobjctypes;      begin        objc_metaclasstype:=tpointerdef(search_named_unit_globaltype('OBJC','POBJC_CLASS',true).typedef);        objc_superclasstype:=tpointerdef(search_named_unit_globaltype('OBJC','POBJC_SUPER',true).typedef);        objc_idtype:=tpointerdef(search_named_unit_globaltype('OBJC','ID',true).typedef);        objc_seltype:=tpointerdef(search_named_unit_globaltype('OBJC','SEL',true).typedef);        objc_objecttype:=trecorddef(search_named_unit_globaltype('OBJC','OBJC_OBJECT',true).typedef);      end;    procedure maybeloadcocoatypes;      var        tsym: ttypesym;        cocoaunit: string[15];      begin        if assigned(objc_fastenumeration) then          exit;        if not(target_info.system in [system_arm_darwin,system_i386_iphonesim,system_aarch64_darwin,system_x86_64_iphonesim]) then          cocoaunit:='COCOAALL'        else          cocoaunit:='IPHONEALL';        tsym:=search_named_unit_globaltype(cocoaunit,'NSFASTENUMERATIONPROTOCOL',false);        if assigned(tsym) then          objc_fastenumeration:=tobjectdef(tsym.typedef)        else          objc_fastenumeration:=nil;        tsym:=search_named_unit_globaltype(cocoaunit,'NSFASTENUMERATIONSTATE',false);        if assigned(tsym) then          objc_fastenumerationstate:=trecorddef(tsym.typedef)        else          objc_fastenumerationstate:=nil;      end;    function use_vectorfpu(def : tdef) : boolean;      begin{$ifdef x86}{$define use_vectorfpuimplemented}        use_vectorfpu:=(is_single(def) and (current_settings.fputype in sse_singlescalar)) or          (is_double(def) and (current_settings.fputype in sse_doublescalar));{$endif x86}{$ifdef arm}{$define use_vectorfpuimplemented}        use_vectorfpu:=(current_settings.fputype in vfp_scalar);{$endif arm}{$ifdef aarch64}{$define use_vectorfpuimplemented}        use_vectorfpu:=true;{$endif aarch64}{$ifndef use_vectorfpuimplemented}        use_vectorfpu:=false;{$endif}      end;end.
 |