| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566 | {    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 }       cutils,cclasses,       { global }       globtype,globals,tokens,       { symtable }       symconst,symbase,symtype,       { ppu }       ppu,       { node }       node,       { aasm }       aasmbase,aasmtai,       cpubase,cpuinfo,       cgbase,cgutils,       parabase       ;    type{************************************************                    TDef************************************************}       tstoreddef = class(tdef)       protected          typesymderef  : tderef;       public          { persistent (available across units) rtti and init tables }          rttitablesym,          inittablesym  : tsym; {trttisym}          rttitablesymderef,          inittablesymderef : tderef;          { local (per module) rtti and init tables }          localrttilab  : array[trttitype] of tasmlabel;          { linked list of global definitions }{$ifdef EXTDEBUG}          fileinfo   : tfileposinfo;{$endif}{$ifdef GDB}          globalnb   : word;          stab_state : tdefstabstatus;{$endif GDB}          constructor create;          constructor ppuloaddef(ppufile:tcompilerppufile);          procedure reset;          function getcopy : tstoreddef;virtual;          procedure ppuwritedef(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);virtual;abstract;          procedure buildderef;override;          procedure buildderefimpl;override;          procedure deref;override;          procedure derefimpl;override;          function  size:aint;override;          function  getvartype:longint;override;          function  alignment:longint;override;          function  is_publishable : boolean;override;          function  needs_inittable : boolean;override;          { debug }{$ifdef GDB}          function get_var_value(const s:string):string;          function stabstr_evaluate(const s:string;const vars:array of string):Pchar;          function  stabstring : pchar;virtual;          procedure concatstabto(asmlist : taasmoutput);virtual;          function  numberstring:string;virtual;          procedure set_globalnb;virtual;          function  allstabstring : pchar;virtual;{$endif GDB}          { rtti generation }          procedure write_rtti_name;          procedure write_rtti_data(rt:trttitype);virtual;          procedure write_child_rtti_data(rt:trttitype);virtual;          function  get_rtti_label(rt:trttitype):tasmsymbol;          { regvars }          function is_intregable : boolean;          function is_fpuregable : boolean;       private          savesize  : aint;       end;       tfiletyp = (ft_text,ft_typed,ft_untyped);       tfiledef = class(tstoreddef)          filetyp : tfiletyp;          typedfiletype : ttype;          constructor createtext;          constructor createuntyped;          constructor createtyped(const tt : ttype);          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          function  gettypename:string;override;          function  getmangledparaname:string;override;          procedure setsize;          { debug }{$ifdef GDB}          function  stabstring : pchar;override;          procedure concatstabto(asmlist : taasmoutput);override;{$endif GDB}       end;       tvariantdef = class(tstoreddef)          varianttype : tvarianttype;          constructor create(v : tvarianttype);          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          function gettypename:string;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure setsize;          function is_publishable : boolean;override;          function needs_inittable : boolean;override;          procedure write_rtti_data(rt:trttitype);override;{$ifdef GDB}          function  numberstring:string;override;          function  stabstring : pchar;override;          procedure concatstabto(asmlist : taasmoutput);override;{$endif GDB}       end;       tformaldef = class(tstoreddef)          constructor create;          constructor ppuload(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          function  gettypename:string;override;{$ifdef GDB}          function  numberstring:string;override;          function  stabstring : pchar;override;          procedure concatstabto(asmlist : taasmoutput);override;{$endif GDB}       end;       tforwarddef = class(tstoreddef)          tosymname : pstring;          forwardpos : tfileposinfo;          constructor create(const s:string;const pos : tfileposinfo);          destructor destroy;override;          function  gettypename:string;override;       end;       terrordef = class(tstoreddef)          constructor create;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function  gettypename:string;override;          function  getmangledparaname : string;override;          { debug }{$ifdef GDB}          function  stabstring : pchar;override;          procedure concatstabto(asmlist : taasmoutput);override;{$endif GDB}       end;       { tpointerdef and tclassrefdef should get a common         base class, but I derived tclassrefdef from tpointerdef         to avoid problems with bugs (FK)       }       tpointerdef = class(tstoreddef)          pointertype : ttype;          is_far : boolean;          constructor create(const tt : ttype);          constructor createfar(const tt : ttype);          function getcopy : tstoreddef;override;          constructor ppuload(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          function  gettypename:string;override;          { debug }{$ifdef GDB}          function  stabstring : pchar;override;          procedure concatstabto(asmlist : taasmoutput);override;{$endif GDB}       end;       Trecord_stabgen_state=record          stabstring:Pchar;          stabsize,staballoc,recoffset:integer;       end;       tabstractrecorddef= class(tstoreddef)       private          Count         : integer;          FRTTIType     : trttitype;{$ifdef GDB}          procedure field_addname(p:Tnamedindexitem;arg:pointer);          procedure field_concatstabto(p:Tnamedindexitem;arg:pointer);{$endif}          procedure count_field_rtti(sym : tnamedindexitem;arg:pointer);          procedure write_field_rtti(sym : tnamedindexitem;arg:pointer);          procedure generate_field_rtti(sym : tnamedindexitem;arg:pointer);       public          symtable : tsymtable;          function  getsymtable(t:tgetsymtable):tsymtable;override;       end;       trecorddef = class(tabstractrecorddef)       public          isunion       : boolean;          constructor create(p : tsymtable);          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy;override;          function getcopy : tstoreddef;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          function  size:aint;override;          function  alignment : longint;override;          function  padalignment: longint;          function  gettypename:string;override;          { debug }{$ifdef GDB}          function  stabstring : pchar;override;          procedure concatstabto(asmlist:taasmoutput);override;{$endif GDB}          function  needs_inittable : boolean;override;          { rtti }          procedure write_child_rtti_data(rt:trttitype);override;          procedure write_rtti_data(rt:trttitype);override;       end;       tprocdef = class;       tobjectdef = class;       timplementedinterfaces = class;       timplintfentry = class(TNamedIndexItem)         intf         : tobjectdef;         intfderef    : tderef;         ioffset      : longint;         implindex    : longint;         namemappings : tdictionary;         procdefs     : TIndexArray;         constructor create(aintf: tobjectdef);         constructor create_deref(const d:tderef);         destructor  destroy; override;       end;       tobjectdef = class(tabstractrecorddef)       private{$ifdef GDB}          procedure proc_addname(p :tnamedindexitem;arg:pointer);          procedure proc_concatstabto(p :tnamedindexitem;arg:pointer);{$endif GDB}          procedure count_published_properties(sym:tnamedindexitem;arg:pointer);          procedure write_property_info(sym : tnamedindexitem;arg:pointer);          procedure generate_published_child_rtti(sym : tnamedindexitem;arg:pointer);          procedure count_published_fields(sym:tnamedindexitem;arg:pointer);          procedure writefields(sym:tnamedindexitem;arg:pointer);       public          childof  : tobjectdef;          childofderef  : tderef;          objname,          objrealname   : pstring;          objectoptions : tobjectoptions;          { to be able to have a variable vmt position }          { and no vmt field for objects without virtuals }          vmt_offset : longint;{$ifdef GDB}          writing_class_record_stab : boolean;{$endif GDB}          objecttype : tobjectdeftype;          iidguid: pguid;          iidstr: pstring;          lastvtableindex: longint;          { store implemented interfaces defs and name mappings }          implementedinterfaces: timplementedinterfaces;          constructor create(ot : tobjectdeftype;const n : string;c : tobjectdef);          constructor ppuload(ppufile:tcompilerppufile);          destructor  destroy;override;          function getcopy : tstoreddef;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function gettypename:string;override;          procedure buildderef;override;          procedure deref;override;          function  getparentdef:tdef;override;          function  size : aint;override;          function  alignment:longint;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  vmt_mangledname : string;          function  rtti_name : string;          procedure check_forwards;          function  is_related(d : tdef) : boolean;override;          function  next_free_name_index : longint;          procedure insertvmt;          procedure set_parent(c : tobjectdef);          function searchdestructor : tprocdef;          { debug }{$ifdef GDB}          function  stabstring : pchar;override;          procedure set_globalnb;override;          function  classnumberstring : string;          procedure concatstabto(asmlist : taasmoutput);override;          function  allstabstring : pchar;override;{$endif GDB}          { rtti }          procedure write_child_rtti_data(rt:trttitype);override;          procedure write_rtti_data(rt:trttitype);override;          function generate_field_table : tasmlabel;       end;       timplementedinterfaces = class          constructor create;          destructor  destroy; override;          function  count: longint;          function  interfaces(intfindex: longint): tobjectdef;          function  interfacesderef(intfindex: longint): tderef;          function  ioffsets(intfindex: longint): longint;          procedure setioffsets(intfindex,iofs:longint);          function  implindex(intfindex:longint):longint;          procedure setimplindex(intfindex,implidx:longint);          function  searchintf(def: tdef): longint;          procedure addintf(def: tdef);          procedure buildderef;          procedure deref;          { add interface reference loaded from ppu }          procedure addintf_deref(const d:tderef;iofs:longint);          procedure clearmappings;          procedure addmappings(intfindex: longint; const name, newname: string);          function  getmappings(intfindex: longint; const name: string; var nextexist: pointer): string;          procedure addimplproc(intfindex: longint; procdef: tprocdef);          function  implproccount(intfindex: longint): longint;          function  implprocs(intfindex: longint; procindex: longint): tprocdef;          function  isimplmergepossible(intfindex, remainindex: longint; var weight: longint): boolean;       private          finterfaces: tindexarray;          procedure checkindex(intfindex: longint);       end;       tclassrefdef = class(tpointerdef)          constructor create(const t:ttype);          constructor ppuload(ppufile:tcompilerppufile);          procedure ppuwrite(ppufile:tcompilerppufile);override;          function gettypename:string;override;          function  is_publishable : boolean;override;          { debug }{$ifdef GDB}          function stabstring : pchar;override;{$endif GDB}       end;       tarraydef = class(tstoreddef)          lowrange,          highrange  : aint;          rangetype  : ttype;          IsConvertedPointer,          IsDynamicArray,          IsVariant,          IsConstructor,          IsArrayOfConst : boolean;       protected          _elementtype : ttype;       public          function elesize : aint;          function elecount : aint;          constructor create_from_pointer(const elemt : ttype);          constructor create(l,h : aint;const t : ttype);          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function  gettypename:string;override;          function  getmangledparaname : string;override;          procedure setelementtype(t: ttype);{$ifdef GDB}          function  stabstring : pchar;override;          procedure concatstabto(asmlist : taasmoutput);override;{$endif GDB}          procedure buildderef;override;          procedure deref;override;          function size : aint;override;          function alignment : longint;override;          { returns the label of the range check string }          function needs_inittable : boolean;override;          procedure write_child_rtti_data(rt:trttitype);override;          procedure write_rtti_data(rt:trttitype);override;          property elementtype : ttype Read _ElementType;       end;       torddef = class(tstoreddef)          low,high : TConstExprInt;          typ      : tbasetype;          constructor create(t : tbasetype;v,b : TConstExprInt);          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function  is_publishable : boolean;override;          function  gettypename:string;override;          procedure setsize;          function getvartype : longint;override;          { debug }{$ifdef GDB}          function  stabstring : pchar;override;{$endif GDB}          { rtti }          procedure write_rtti_data(rt:trttitype);override;       end;       tfloatdef = class(tstoreddef)          typ : tfloattype;          constructor create(t : tfloattype);          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function  gettypename:string;override;          function  is_publishable : boolean;override;          procedure setsize;          function  getvartype:longint;override;          { debug }{$ifdef GDB}          function stabstring : pchar;override;          procedure concatstabto(asmlist:taasmoutput);override;{$endif GDB}          { rtti }          procedure write_rtti_data(rt:trttitype);override;       end;       tabstractprocdef = class(tstoreddef)          { saves a definition to the return type }          rettype         : ttype;          parast          : tsymtable;          paras           : tparalist;          proctypeoption  : tproctypeoption;          proccalloption  : tproccalloption;          procoptions     : tprocoptions;          requiredargarea : aint;          { number of user visibile parameters }          maxparacount,          minparacount    : byte;{$ifdef i386}          fpu_used        : longint;    { how many stack fpu must be empty }{$endif i386}          funcretloc : array[tcallercallee] of TLocation;          has_paraloc_info : boolean; { paraloc info is available }          constructor create(level:byte);          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy;override;          procedure  ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          procedure releasemem;          procedure calcparas;          function  typename_paras(showhidden:boolean): string;          procedure test_if_fpu_result;          function  is_methodpointer:boolean;virtual;          function  is_addressonly:boolean;virtual;          { debug }{$ifdef GDB}          function  stabstring : pchar;override;{$endif GDB}       private          procedure count_para(p:tnamedindexitem;arg:pointer);          procedure insert_para(p:tnamedindexitem;arg:pointer);       end;       tprocvardef = class(tabstractprocdef)          constructor create(level:byte);          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          function  getsymtable(t:tgetsymtable):tsymtable;override;          function  size : aint;override;          function  gettypename:string;override;          function  is_publishable : boolean;override;          function  is_methodpointer:boolean;override;          function  is_addressonly:boolean;override;          function  getmangledparaname:string;override;          { debug }{$ifdef GDB}          function stabstring : pchar;override;          procedure concatstabto(asmlist:taasmoutput);override;{$endif GDB}          { rtti }          procedure write_rtti_data(rt:trttitype);override;       end;       tmessageinf = record         case integer of           0 : (str : pchar);           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}       tprocdef = class(tabstractprocdef)       private          _mangledname : pstring;{$ifdef GDB}          isstabwritten : boolean;{$endif GDB}       public          extnumber      : word;          messageinf : tmessageinf;{$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;          { symbol owning this definition }          procsym : tsym;          procsymderef : tderef;          { alias names }          aliasnames : tstringlist;          { symtables }          localst : tsymtable;          funcretsym : tsym;          funcretsymderef : tderef;          { browser info }          lastref,          defref,          lastwritten : tref;          refcount : longint;          _class : tobjectdef;          _classderef : tderef;{$ifdef powerpc}          { library symbol for AmigaOS/MorphOS }          libsym : tsym;          libsymderef : tderef;{$endif powerpc}          { name of the result variable to insert in the localsymtable }          resultname : stringid;          { true, if the procedure is only declared            (forward procedure) }          forwarddef,          { true if the procedure is declared in the interface }          interfacedef : boolean;          { true if the procedure has a forward declaration }          hasforward : boolean;          { import info }          import_dll,          import_name : pstring;          import_nr   : word;          { info for inlining the subroutine, if this pointer is nil,            the procedure can't be inlined }          inlininginfo : pinlininginfo;{$ifdef oldregvars}          regvarinfo: pregvarinfo;{$endif oldregvars}          constructor create(level:byte);          constructor ppuload(ppufile:tcompilerppufile);          destructor  destroy;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure buildderefimpl;override;          procedure deref;override;          procedure derefimpl;override;          function  getsymtable(t:tgetsymtable):tsymtable;override;          function gettypename : string;override;          function  mangledname : string;          procedure setmangledname(const s : string);          procedure load_references(ppufile:tcompilerppufile;locals:boolean);          function  write_references(ppufile:tcompilerppufile;locals:boolean):boolean;          { inserts the local symbol table, if this is not            no local symbol table is built. Should be called only            when we are sure that a local symbol table will be required.          }          procedure insert_localst;          function  fullprocname(showhidden:boolean):string;          function  cplusplusmangledname : string;          function  is_methodpointer:boolean;override;          function  is_addressonly:boolean;override;          function  is_visible_for_object(currobjdef:tobjectdef):boolean;          { debug }{$ifdef GDB}          function  numberstring:string;override;          function  stabstring : pchar;override;          procedure concatstabto(asmlist : taasmoutput);override;{$endif GDB}       end;       { single linked list of overloaded procs }       pprocdeflist = ^tprocdeflist;       tprocdeflist = record         def  : tprocdef;         defderef : tderef;         next : pprocdeflist;       end;       tstringdef = class(tstoreddef)          string_typ : tstringtype;          len        : aint;          constructor createshort(l : byte);          constructor loadshort(ppufile:tcompilerppufile);          constructor createlong(l : aint);          constructor loadlong(ppufile:tcompilerppufile);       {$ifdef ansistring_bits}          constructor createansi(l:aint;bits:Tstringbits);          constructor loadansi(ppufile:tcompilerppufile;bits:Tstringbits);       {$else}          constructor createansi(l : aint);          constructor loadansi(ppufile:tcompilerppufile);       {$endif}          constructor createwide(l : aint);          constructor loadwide(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;          function  stringtypname:string;          procedure ppuwrite(ppufile:tcompilerppufile);override;          function  gettypename:string;override;          function  getmangledparaname:string;override;          function  is_publishable : boolean;override;          { debug }{$ifdef GDB}          function  stabstring : pchar;override;          procedure concatstabto(asmlist : taasmoutput);override;{$endif GDB}          function alignment : longint;override;          { init/final }          function  needs_inittable : boolean;override;          { rtti }          procedure write_rtti_data(rt:trttitype);override;       end;       tenumdef = class(tstoreddef)          minval,          maxval    : aint;          has_jumps : boolean;          firstenum : tsym;  {tenumsym}          basedef   : tenumdef;          basedefderef : tderef;          constructor create;          constructor create_subrange(_basedef:tenumdef;_min,_max:aint);          constructor ppuload(ppufile:tcompilerppufile);          destructor destroy;override;          function getcopy : tstoreddef;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          procedure derefimpl;override;          function  gettypename:string;override;          function  is_publishable : boolean;override;          procedure calcsavesize;          procedure setmax(_max:aint);          procedure setmin(_min:aint);          function  min:aint;          function  max:aint;          { debug }{$ifdef GDB}          function stabstring : pchar;override;{$endif GDB}          { rtti }          procedure write_rtti_data(rt:trttitype);override;          procedure write_child_rtti_data(rt:trttitype);override;       private          procedure correct_owner_symtable;       end;       tsetdef = class(tstoreddef)          elementtype : ttype;          settype : tsettype;          setbase,          setmax : aint;          constructor create(const t:ttype;high : aint);          constructor ppuload(ppufile:tcompilerppufile);          destructor  destroy;override;          function getcopy : tstoreddef;override;          procedure ppuwrite(ppufile:tcompilerppufile);override;          procedure buildderef;override;          procedure deref;override;          function  gettypename:string;override;          function  is_publishable : boolean;override;          { debug }{$ifdef GDB}          function  stabstring : pchar;override;          procedure concatstabto(asmlist : taasmoutput);override;{$endif GDB}          { rtti }          procedure write_rtti_data(rt:trttitype);override;          procedure write_child_rtti_data(rt:trttitype);override;       end;       Tdefmatch=(dm_exact,dm_equal,dm_convertl1);    var       aktobjectdef : tobjectdef;  { used for private functions check !! }{$ifdef GDB}       writing_def_stabs : boolean;       { for STAB debugging }       globaltypecount  : word;       pglobaltypecount : pword;{$endif GDB}    { default types }       generrortype,              { error in definition }       voidpointertype,           { pointer for Void-Pointerdef }       charpointertype,           { pointer for Char-Pointerdef }       widecharpointertype,       { pointer for WideChar-Pointerdef }       voidfarpointertype,       cformaltype,               { unique formal definition }       voidtype,                  { Void (procedure) }       cchartype,                 { Char }       cwidechartype,             { WideChar }       booltype,                  { boolean type }       u8inttype,                 { 8-Bit unsigned integer }       s8inttype,                 { 8-Bit signed integer }       u16inttype,                { 16-Bit unsigned integer }       s16inttype,                { 16-Bit signed integer }       u32inttype,                { 32-Bit unsigned integer }       s32inttype,                { 32-Bit signed integer }       u64inttype,                { 64-bit unsigned integer }       s64inttype,                { 64-bit signed integer }       s32floattype,              { pointer for realconstn }       s64floattype,              { pointer for realconstn }       s80floattype,              { pointer to type of temp. floats }       s64currencytype,           { pointer to a currency type }       cshortstringtype,          { pointer to type of short string const   }       clongstringtype,           { pointer to type of long string const   }{$ifdef ansistring_bits}       cansistringtype16,         { pointer to type of ansi string const  }       cansistringtype32,         { pointer to type of ansi string const  }       cansistringtype64,         { pointer to type of ansi string const  }{$else}       cansistringtype,           { pointer to type of ansi string const  }{$endif}       cwidestringtype,           { pointer to type of wide string const  }       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 }       { we use only one variant def for every variant class }       cvarianttype,       colevarianttype,       { default integer type s32inttype on 32 bit systems, s64bittype on 64 bit systems }       sinttype,       uinttype,       { unsigned ord type with the same size as a pointer }       ptrinttype,       { several types to simulate more or less C++ objects for GDB }       vmttype,       vmtarraytype,       pvmttype      : ttype;     { 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 TGUID type         of all interfaces         }       rec_tguid : trecorddef;    const{$ifdef i386}       pbestrealtype : ^ttype = @s80floattype;{$endif}{$ifdef x86_64}       pbestrealtype : ^ttype = @s80floattype;{$endif}{$ifdef m68k}       pbestrealtype : ^ttype = @s64floattype;{$endif}{$ifdef alpha}       pbestrealtype : ^ttype = @s64floattype;{$endif}{$ifdef powerpc}       pbestrealtype : ^ttype = @s64floattype;{$endif}{$ifdef ia64}       pbestrealtype : ^ttype = @s64floattype;{$endif}{$ifdef SPARC}       pbestrealtype : ^ttype = @s64floattype;{$endif SPARC}{$ifdef vis}       pbestrealtype : ^ttype = @s64floattype;{$endif vis}{$ifdef ARM}       pbestrealtype : ^ttype = @s64floattype;{$endif ARM}{$ifdef MIPS}       pbestrealtype : ^ttype = @s64floattype;{$endif MIPS}    function make_mangledname(const typeprefix:string;st:tsymtable;const suffix:string):string;    { should be in the types unit, but the types unit uses the node stuff :( }    function is_interfacecom(def: tdef): boolean;    function is_interfacecorba(def: tdef): boolean;    function is_interface(def: tdef): boolean;    function is_object(def: tdef): boolean;    function is_class(def: tdef): boolean;    function is_cppclass(def: tdef): boolean;    function is_class_or_interface(def: tdef): boolean;implementation    uses      strings,      { global }      verbose,      { target }      systems,aasmcpu,paramgr,      { symtable }      symsym,symtable,symutil,defutil,      { module }{$ifdef GDB}      gdb,{$endif GDB}      fmodule,      { other }      gendef,      crc      ;{****************************************************************************                                  Constants****************************************************************************}    const      varempty = 0;      varnull = 1;      varsmallint = 2;      varinteger = 3;      varsingle = 4;      vardouble = 5;      varcurrency = 6;      vardate = 7;      varolestr = 8;      vardispatch = 9;      varerror = 10;      varboolean = 11;      varvariant = 12;      varunknown = 13;      vardecimal = 14;      varshortint = 16;      varbyte = 17;      varword = 18;      varlongword = 19;      varint64 = 20;      varqword = 21;      varUndefined = -1;      varstrarg = $48;      varstring = $100;      varany = $101;      vartypemask = $fff;      vararray = $2000;      varbyref = $4000;{****************************************************************************                                  Helpers****************************************************************************}    function make_mangledname(const typeprefix:string;st:tsymtable;const suffix:string):string;      var        s,hs,        prefix : string;        oldlen,        newlen,        i   : longint;        crc : dword;        hp  : tparavarsym;      begin        prefix:='';        if not assigned(st) then         internalerror(200204212);        { sub procedures }        while (st.symtabletype=localsymtable) do         begin           if st.defowner.deftype<>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;           oldlen:=length(s);           for i:=0 to tprocdef(st.defowner).paras.count-1 do            begin              hp:=tparavarsym(tprocdef(st.defowner).paras[i]);              if not(vo_is_hidden_para in hp.varoptions) then                s:=s+'$'+hp.vartype.def.mangledparaname;            end;           if not is_void(tprocdef(st.defowner).rettype.def) then             s:=s+'$$'+tprocdef(st.defowner).rettype.def.mangledparaname;           newlen:=length(s);           { Replace with CRC if the parameter line is very long }           if (newlen-oldlen>12) and              ((newlen>128) or (newlen-oldlen>64)) then             begin               crc:=$ffffffff;               for i:=0 to tprocdef(st.defowner).paras.count-1 do                 begin                   hp:=tparavarsym(tprocdef(st.defowner).paras[i]);                   if not(vo_is_hidden_para in hp.varoptions) then                     begin                       hs:=hp.vartype.def.mangledparaname;                       crc:=UpdateCrc32(crc,hs[1],length(hs));                     end;                 end;               hs:=hp.vartype.def.mangledparaname;               crc:=UpdateCrc32(crc,hs[1],length(hs));               s:=Copy(s,1,oldlen)+'$crc'+hexstr(crc,8);             end;           if prefix<>'' then             prefix:=s+'_'+prefix           else             prefix:=s;           st:=st.defowner.owner;         end;        { object/classes symtable }        if (st.symtabletype=objectsymtable) then         begin           if st.defowner.deftype<>objectdef then            internalerror(200204174);           prefix:=tobjectdef(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);        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 }        if (target_info.system = system_powerpc_darwin) and           (result[1] = 'L') then          result := '_' + result;      end;{****************************************************************************                     TDEF (base class for definitions)****************************************************************************}    constructor tstoreddef.create;      begin         inherited create;         savesize := 0;{$ifdef EXTDEBUG}         fileinfo := aktfilepos;{$endif}         if registerdef then           symtablestack.registerdef(self);{$ifdef GDB}         stab_state:=stab_state_unused;         globalnb := 0;{$endif GDB}         fillchar(localrttilab,sizeof(localrttilab),0);      end;    constructor tstoreddef.ppuloaddef(ppufile:tcompilerppufile);      begin         inherited create;{$ifdef EXTDEBUG}         fillchar(fileinfo,sizeof(fileinfo),0);{$endif}{$ifdef GDB}         stab_state:=stab_state_unused;         globalnb := 0;{$endif GDB}         fillchar(localrttilab,sizeof(localrttilab),0);      { load }         indexnr:=ppufile.getword;         ppufile.getderef(typesymderef);         ppufile.getsmallset(defoptions);         if df_has_rttitable in defoptions then          ppufile.getderef(rttitablesymderef);         if df_has_inittable in defoptions then          ppufile.getderef(inittablesymderef);      end;    procedure Tstoreddef.reset;      begin{$ifdef GDB}        stab_state:=stab_state_unused;{$endif GDB}        if assigned(rttitablesym) then          trttisym(rttitablesym).lab := nil;        if assigned(inittablesym) then          trttisym(inittablesym).lab := nil;        localrttilab[initrtti]:=nil;        localrttilab[fullrtti]:=nil;      end;    function tstoreddef.getcopy : tstoreddef;      begin        Message(sym_e_cant_create_unique_type);        getcopy:=terrordef.create;      end;    procedure tstoreddef.ppuwritedef(ppufile:tcompilerppufile);      begin        ppufile.putword(indexnr);        ppufile.putderef(typesymderef);        ppufile.putsmallset(defoptions);        if df_has_rttitable in defoptions then         ppufile.putderef(rttitablesymderef);        if df_has_inittable in defoptions then         ppufile.putderef(inittablesymderef);{$ifdef GDB}        if globalnb=0 then          begin            if (cs_gdb_dbx in aktglobalswitches) and               assigned(owner) then              globalnb := owner.getnewtypecount            else              set_globalnb;          end;{$endif GDB}      end;    procedure tstoreddef.buildderef;      begin        typesymderef.build(typesym);        rttitablesymderef.build(rttitablesym);        inittablesymderef.build(inittablesym);      end;    procedure tstoreddef.buildderefimpl;      begin      end;    procedure tstoreddef.deref;      begin        typesym:=ttypesym(typesymderef.resolve);        if df_has_rttitable in defoptions then          rttitablesym:=trttisym(rttitablesymderef.resolve);        if df_has_inittable in defoptions then          inittablesym:=trttisym(inittablesymderef.resolve);      end;    procedure tstoreddef.derefimpl;      begin      end;    function tstoreddef.size : aint;      begin         size:=savesize;      end;    function tstoreddef.getvartype:longint;      begin        result:=varUndefined;      end;    function tstoreddef.alignment : longint;      begin         { natural alignment by default }         alignment:=size_2_align(savesize);      end;{$ifdef GDB}    procedure tstoreddef.set_globalnb;      begin        globalnb:=PGlobalTypeCount^;        inc(PglobalTypeCount^);      end;    function Tstoreddef.get_var_value(const s:string):string;      begin        if s='numberstring' then          get_var_value:=numberstring        else if s='sym_name' then          if assigned(typesym) then             get_var_value:=Ttypesym(typesym).name          else             get_var_value:=' '        else if s='N_LSYM' then          get_var_value:=tostr(N_LSYM)        else if s='savesize' then          get_var_value:=tostr(savesize);      end;    function Tstoreddef.stabstr_evaluate(const s:string;const vars:array of string):Pchar;      begin        stabstr_evaluate:=string_evaluate(s,@get_var_value,vars);      end;    function tstoreddef.stabstring : pchar;      begin        stabstring:=stabstr_evaluate('t${numberstring};',[]);      end;    function tstoreddef.numberstring : string;      begin        { Stab must already be written, or we must be busy writing it }        if writing_def_stabs and           not(stab_state in [stab_state_writing,stab_state_written]) then          internalerror(200403091);        { Keep track of used stabs, this info is only usefull for stabs          referenced by the symbols. Definitions will always include all          required stabs }        if stab_state=stab_state_unused then          stab_state:=stab_state_used;        { Need a new number? }        if globalnb=0 then          begin            if (cs_gdb_dbx in aktglobalswitches) and               assigned(owner) then              globalnb := owner.getnewtypecount            else              set_globalnb;          end;        if (cs_gdb_dbx in aktglobalswitches) and           assigned(typesym) and           (ttypesym(typesym).owner.symtabletype in [staticsymtable,globalsymtable]) and           (ttypesym(typesym).owner.iscurrentunit) then          result:='('+tostr(tabstractunitsymtable(ttypesym(typesym).owner).moduleid)+','+tostr(tstoreddef(ttypesym(typesym).restype.def).globalnb)+')'        else          result:=tostr(globalnb);      end;    function tstoreddef.allstabstring : pchar;      var        stabchar : string[2];        ss,st,su : pchar;      begin        ss := stabstring;        stabchar := 't';        if deftype in tagtypes then          stabchar := 'Tt';        { Here we maybe generate a type, so we have to use numberstring }        st:=stabstr_evaluate('"${sym_name}:$1$2=',[stabchar,numberstring]);        reallocmem(st,strlen(ss)+512);        { line info is set to 0 for all defs, because the def can be in an other          unit and then the linenumber is invalid in the current sourcefile }        su:=stabstr_evaluate('",${N_LSYM},0,0,0',[]);        strcopy(strecopy(strend(st),ss),su);        reallocmem(st,strlen(st)+1);        allstabstring:=st;        strdispose(ss);        strdispose(su);      end;    procedure tstoreddef.concatstabto(asmlist : taasmoutput);      var        stab_str : pchar;      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;        If cs_gdb_dbx in aktglobalswitches then          begin            { otherwise you get two of each def }            If assigned(typesym) then              begin                if (ttypesym(typesym).owner = nil) or                   ((ttypesym(typesym).owner.symtabletype = globalsymtable) and                    tglobalsymtable(ttypesym(typesym).owner).dbx_count_ok)  then                  begin                    {with DBX we get the definition from the other objects }                    stab_state := stab_state_written;                    exit;                  end;              end;          end;        { to avoid infinite loops }        stab_state := stab_state_writing;        stab_str := allstabstring;        asmList.concat(Tai_stabs.Create(stab_str));        stab_state := stab_state_written;      end;{$endif GDB}    procedure tstoreddef.write_rtti_name;      var         str : string;      begin         { name }         if assigned(typesym) then           begin              str:=ttypesym(typesym).realname;              rttiList.concat(Tai_string.Create(chr(length(str))+str));           end         else           rttiList.concat(Tai_string.Create(#0))      end;    procedure tstoreddef.write_rtti_data(rt:trttitype);      begin        rttilist.concat(tai_const.create_8bit(tkUnknown));        write_rtti_name;      end;    procedure tstoreddef.write_child_rtti_data(rt:trttitype);      begin      end;    function tstoreddef.get_rtti_label(rt:trttitype) : tasmsymbol;      begin         { try to reuse persistent rtti data }         if (rt=fullrtti) and (df_has_rttitable in defoptions) then          get_rtti_label:=trttisym(rttitablesym).get_label         else          if (rt=initrtti) and (df_has_inittable in defoptions) then           get_rtti_label:=trttisym(inittablesym).get_label         else          begin            if not assigned(localrttilab[rt]) then             begin               objectlibrary.getdatalabel(localrttilab[rt]);               write_child_rtti_data(rt);               maybe_new_object_file(rttiList);               new_section(rttiList,sec_rodata,localrttilab[rt].name,const_align(sizeof(aint)));               rttiList.concat(Tai_symbol.Create_global(localrttilab[rt],0));               write_rtti_data(rt);               rttiList.concat(Tai_symbol_end.Create(localrttilab[rt]));             end;            get_rtti_label:=localrttilab[rt];          end;      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.is_intregable : boolean;     begin        is_intregable:=false;        case deftype of          orddef,          pointerdef,          enumdef:            is_intregable:=true;          procvardef :            is_intregable:=not(po_methodpointer in tprocvardef(self).procoptions);          objectdef:            is_intregable:=is_class(self) or is_interface(self);          setdef:            is_intregable:=(tsetdef(self).settype=smallset);        end;     end;   function tstoreddef.is_fpuregable : boolean;     begin{$ifdef x86}       result:=false;{$else x86}       result:=(deftype=floatdef);{$endif x86}     end;{****************************************************************************                               Tstringdef****************************************************************************}    constructor tstringdef.createshort(l : byte);      begin         inherited create;         string_typ:=st_shortstring;         deftype:=stringdef;         len:=l;         savesize:=len+1;      end;    constructor tstringdef.loadshort(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         string_typ:=st_shortstring;         deftype:=stringdef;         len:=ppufile.getbyte;         savesize:=len+1;      end;    constructor tstringdef.createlong(l : aint);      begin         inherited create;         string_typ:=st_longstring;         deftype:=stringdef;         len:=l;         savesize:=sizeof(aint);      end;    constructor tstringdef.loadlong(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=stringdef;         string_typ:=st_longstring;         len:=ppufile.getaint;         savesize:=sizeof(aint);      end;{$ifdef ansistring_bits}    constructor tstringdef.createansi(l:aint;bits:Tstringbits);      begin         inherited create;         case bits of           sb_16:             string_typ:=st_ansistring16;           sb_32:             string_typ:=st_ansistring32;           sb_64:             string_typ:=st_ansistring64;         end;         deftype:=stringdef;         len:=l;         savesize:=POINTER_SIZE;      end;    constructor tstringdef.loadansi(ppufile:tcompilerppufile;bits:Tstringbits);      begin         inherited ppuloaddef(ppufile);         deftype:=stringdef;         case bits of           sb_16:             string_typ:=st_ansistring16;           sb_32:             string_typ:=st_ansistring32;           sb_64:             string_typ:=st_ansistring64;         end;         len:=ppufile.getaint;         savesize:=POINTER_SIZE;      end;{$else}    constructor tstringdef.createansi(l:aint);      begin         inherited create;         string_typ:=st_ansistring;         deftype:=stringdef;         len:=l;         savesize:=sizeof(aint);      end;    constructor tstringdef.loadansi(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=stringdef;         string_typ:=st_ansistring;         len:=ppufile.getaint;         savesize:=sizeof(aint);      end;{$endif}    constructor tstringdef.createwide(l : aint);      begin         inherited create;         string_typ:=st_widestring;         deftype:=stringdef;         len:=l;         savesize:=sizeof(aint);      end;    constructor tstringdef.loadwide(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=stringdef;         string_typ:=st_widestring;         len:=ppufile.getaint;         savesize:=sizeof(aint);      end;    function tstringdef.getcopy : tstoreddef;      begin        result:=tstringdef.create;        result.deftype:=stringdef;        tstringdef(result).string_typ:=string_typ;        tstringdef(result).len:=len;        tstringdef(result).savesize:=savesize;      end;    function tstringdef.stringtypname:string;{$ifdef ansistring_bits}      const        typname:array[tstringtype] of string[9]=('',          'shortstr','longstr','ansistr16','ansistr32','ansistr64','widestr'        );{$else}      const        typname:array[tstringtype] of string[8]=('',          'shortstr','longstr','ansistr','widestr'        );{$endif}      begin        stringtypname:=typname[string_typ];      end;    procedure tstringdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         if string_typ=st_shortstring then           begin{$ifdef extdebug}            if len > 255 then internalerror(12122002);{$endif}            ppufile.putbyte(byte(len))           end         else           ppufile.putaint(len);         case string_typ of            st_shortstring : ppufile.writeentry(ibshortstringdef);            st_longstring : ppufile.writeentry(iblongstringdef);         {$ifdef ansistring_bits}            st_ansistring16 : ppufile.writeentry(ibansistring16def);            st_ansistring32 : ppufile.writeentry(ibansistring32def);            st_ansistring64 : ppufile.writeentry(ibansistring64def);         {$else}            st_ansistring : ppufile.writeentry(ibansistringdef);         {$endif}            st_widestring : ppufile.writeentry(ibwidestringdef);         end;      end;{$ifdef GDB}    function tstringdef.stabstring : pchar;      var        bytest,charst,longst : string;        slen : aint;      begin        case string_typ of           st_shortstring:             begin               charst:=tstoreddef(cchartype.def).numberstring;               { this is what I found in stabs.texinfo but                 gdb 4.12 for go32 doesn't understand that !! }             {$IfDef GDBknowsstrings}                stabstring:=stabstr_evaluate('n$1;$2',[charst,tostr(len)]);             {$else}               { fix length of openshortstring }               slen:=len;               if slen=0 then                 slen:=255;               bytest:=tstoreddef(u8inttype.def).numberstring;               stabstring:=stabstr_evaluate('s$1length:$2,0,8;st:ar$2;1;$3;$4,8,$5;;',                           [tostr(slen+1),bytest,tostr(slen),charst,tostr(slen*8)]);             {$EndIf}             end;           st_longstring:             begin               charst:=tstoreddef(cchartype.def).numberstring;               { this is what I found in stabs.texinfo but                 gdb 4.12 for go32 doesn't understand that !! }             {$IfDef GDBknowsstrings}               stabstring:=stabstr_evaluate('n$1;$2',[charst,tostr(len)]);             {$else}               bytest:=tstoreddef(u8inttype.def).numberstring;               longst:=tstoreddef(u32inttype.def).numberstring;               stabstring:=stabstr_evaluate('s$1length:$2,0,32;dummy:$6,32,8;st:ar$2;1;$3;$4,40,$5;;',                            [tostr(len+5),longst,tostr(len),charst,tostr(len*8),bytest]);              {$EndIf}             end;         {$ifdef ansistring_bits}           st_ansistring16,st_ansistring32,st_ansistring64:         {$else}           st_ansistring:         {$endif}             begin               { an ansi string looks like a pchar easy !! }               charst:=tstoreddef(cchartype.def).numberstring;               stabstring:=strpnew('*'+charst);             end;           st_widestring:             begin               { an ansi string looks like a pwidechar easy !! }               charst:=tstoreddef(cwidechartype.def).numberstring;               stabstring:=strpnew('*'+charst);             end;        end;      end;    procedure tstringdef.concatstabto(asmlist:taasmoutput);      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;        case string_typ of           st_shortstring:             begin               tstoreddef(cchartype.def).concatstabto(asmlist);             {$IfNDef GDBknowsstrings}               tstoreddef(u8inttype.def).concatstabto(asmlist);             {$EndIf}             end;           st_longstring:             begin               tstoreddef(cchartype.def).concatstabto(asmlist);             {$IfNDef GDBknowsstrings}               tstoreddef(u8inttype.def).concatstabto(asmlist);               tstoreddef(u32inttype.def).concatstabto(asmlist);             {$EndIf}             end;         {$ifdef ansistring_bits}           st_ansistring16,st_ansistring32,st_ansistring64:         {$else}           st_ansistring:         {$endif}             tstoreddef(cchartype.def).concatstabto(asmlist);           st_widestring:             tstoreddef(cwidechartype.def).concatstabto(asmlist);        end;        inherited concatstabto(asmlist);      end;{$endif GDB}    function tstringdef.needs_inittable : boolean;      begin      {$ifdef ansistring_bits}         needs_inittable:=string_typ in [st_ansistring16,st_ansistring32,st_ansistring64,st_widestring];      {$else}         needs_inittable:=string_typ in [st_ansistring,st_widestring];      {$endif}      end;    function tstringdef.gettypename : string;{$ifdef ansistring_bits}      const         names : array[tstringtype] of string[20] = ('',           'shortstring','longstring','ansistring16','ansistring32','ansistring64','widestring');{$else}      const         names : array[tstringtype] of string[20] = ('',           'ShortString','LongString','AnsiString','WideString');{$endif}      begin         gettypename:=names[string_typ];      end;    function tstringdef.alignment : longint;      begin        case string_typ of          st_widestring,          st_ansistring:            alignment:=size_2_align(savesize);          st_longstring,          st_shortstring:            alignment:=size_2_align(1);          else            internalerror(200412301);        end;      end;    procedure tstringdef.write_rtti_data(rt:trttitype);      begin         case string_typ of          {$ifdef ansistring_bits}            st_ansistring16:              begin                 rttiList.concat(Tai_const.Create_8bit(tkA16String));                 write_rtti_name;              end;            st_ansistring32:              begin                 rttiList.concat(Tai_const.Create_8bit(tkA32String));                 write_rtti_name;              end;            st_ansistring64:              begin                 rttiList.concat(Tai_const.Create_8bit(tkA64String));                 write_rtti_name;              end;          {$else}            st_ansistring:              begin                 rttiList.concat(Tai_const.Create_8bit(tkAString));                 write_rtti_name;              end;          {$endif}            st_widestring:              begin                 rttiList.concat(Tai_const.Create_8bit(tkWString));                 write_rtti_name;              end;            st_longstring:              begin                 rttiList.concat(Tai_const.Create_8bit(tkLString));                 write_rtti_name;              end;            st_shortstring:              begin                 rttiList.concat(Tai_const.Create_8bit(tkSString));                 write_rtti_name;                 rttiList.concat(Tai_const.Create_8bit(len));{$ifdef cpurequiresproperalignment}                 rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}              end;         end;      end;    function tstringdef.getmangledparaname : string;      begin        getmangledparaname:='STRING';      end;    function tstringdef.is_publishable : boolean;      begin         is_publishable:=true;      end;{****************************************************************************                                 TENUMDEF****************************************************************************}    constructor tenumdef.create;      begin         inherited create;         deftype:=enumdef;         minval:=0;         maxval:=0;         calcsavesize;         has_jumps:=false;         basedef:=nil;         firstenum:=nil;         correct_owner_symtable;      end;    constructor tenumdef.create_subrange(_basedef:tenumdef;_min,_max:aint);      begin         inherited create;         deftype:=enumdef;         minval:=_min;         maxval:=_max;         basedef:=_basedef;         calcsavesize;         has_jumps:=false;         firstenum:=basedef.firstenum;         while assigned(firstenum) and (tenumsym(firstenum).value<>minval) do           firstenum:=tenumsym(firstenum).nextenum;         correct_owner_symtable;      end;    constructor tenumdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=enumdef;         ppufile.getderef(basedefderef);         minval:=ppufile.getaint;         maxval:=ppufile.getaint;         savesize:=ppufile.getaint;         has_jumps:=false;         firstenum:=Nil;      end;    function tenumdef.getcopy : tstoreddef;      begin        if assigned(basedef) then          result:=tenumdef.create_subrange(basedef,minval,maxval)        else          begin            result:=tenumdef.create;            tenumdef(result).minval:=minval;            tenumdef(result).maxval:=maxval;          end;        tenumdef(result).has_jumps:=has_jumps;        tenumdef(result).firstenum:=firstenum;        tenumdef(result).basedefderef:=basedefderef;      end;    procedure tenumdef.calcsavesize;      begin        if (aktpackenum=8) or (min<low(longint)) or (int64(max)>high(cardinal)) then         savesize:=8        else         if (aktpackenum=4) or (min<low(smallint)) or (max>high(word)) then          savesize:=4        else         if (aktpackenum=2) or (min<low(shortint)) or (max>high(byte)) then          savesize:=2        else         savesize:=1;      end;    procedure tenumdef.setmax(_max:aint);      begin        maxval:=_max;        calcsavesize;      end;    procedure tenumdef.setmin(_min:aint);      begin        minval:=_min;        calcsavesize;      end;    function tenumdef.min:aint;      begin        min:=minval;      end;    function tenumdef.max:aint;      begin        max:=maxval;      end;    procedure tenumdef.buildderef;      begin        inherited buildderef;        basedefderef.build(basedef);      end;    procedure tenumdef.deref;      begin        inherited deref;        basedef:=tenumdef(basedefderef.resolve);        { restart ordering }        firstenum:=nil;      end;    procedure tenumdef.derefimpl;      begin        if assigned(basedef) and           (firstenum=nil) then          begin            firstenum:=basedef.firstenum;            while assigned(firstenum) and (tenumsym(firstenum).value<>minval) do              firstenum:=tenumsym(firstenum).nextenum;          end;      end;    destructor tenumdef.destroy;      begin        inherited destroy;      end;    procedure tenumdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.putderef(basedefderef);         ppufile.putaint(min);         ppufile.putaint(max);         ppufile.putaint(savesize);         ppufile.writeentry(ibenumdef);      end;    { used for enumdef because the symbols are      inserted in the owner symtable }    procedure tenumdef.correct_owner_symtable;      var         st : tsymtable;      begin         if assigned(owner) and            (owner.symtabletype in [recordsymtable,objectsymtable]) then           begin              owner.defindex.deleteindex(self);              st:=owner;              while (st.symtabletype in [recordsymtable,objectsymtable]) do                st:=st.next;              st.registerdef(self);           end;      end;{$ifdef GDB}    function tenumdef.stabstring : pchar;    var st:Pchar;        p:Tenumsym;        s:string;        memsize,stl:cardinal;    begin      memsize:=memsizeinc;      getmem(st,memsize);      { we can specify the size with @s<size>; prefix PM }      if savesize <> std_param_align then        strpcopy(st,'@s'+tostr(savesize*8)+';e')      else        strpcopy(st,'e');      p := tenumsym(firstenum);      stl:=strlen(st);      while assigned(p) do        begin          s :=p.name+':'+tostr(p.value)+',';          { place for the ending ';' also }          if (stl+length(s)+1>=memsize) then            begin              inc(memsize,memsizeinc);              reallocmem(st,memsize);            end;          strpcopy(st+stl,s);          inc(stl,length(s));          p:=p.nextenum;        end;      st[stl]:=';';      st[stl+1]:=#0;      reallocmem(st,stl+2);      stabstring:=st;    end;{$endif GDB}    procedure tenumdef.write_child_rtti_data(rt:trttitype);      begin         if assigned(basedef) then           basedef.get_rtti_label(rt);      end;    procedure tenumdef.write_rtti_data(rt:trttitype);      var         hp : tenumsym;      begin         rttiList.concat(Tai_const.Create_8bit(tkEnumeration));         write_rtti_name;{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}         case longint(savesize) of            1:              rttiList.concat(Tai_const.Create_8bit(otUByte));            2:              rttiList.concat(Tai_const.Create_8bit(otUWord));            4:              rttiList.concat(Tai_const.Create_8bit(otULong));         end;{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(4));{$endif cpurequiresproperalignment}         rttiList.concat(Tai_const.Create_32bit(min));         rttiList.concat(Tai_const.Create_32bit(max));         if assigned(basedef) then           rttiList.concat(Tai_const.Create_sym(basedef.get_rtti_label(rt)))         else           rttiList.concat(Tai_const.create_sym(nil));         hp:=tenumsym(firstenum);         while assigned(hp) do           begin              rttiList.concat(Tai_const.Create_8bit(length(hp.realname)));              rttiList.concat(Tai_string.Create(hp.realname));              hp:=hp.nextenum;           end;         rttiList.concat(Tai_const.Create_8bit(0));      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 : tbasetype;v,b : TConstExprInt);      begin         inherited create;         deftype:=orddef;         low:=v;         high:=b;         typ:=t;         setsize;      end;    constructor torddef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=orddef;         typ:=tbasetype(ppufile.getbyte);         if sizeof(TConstExprInt)=8 then           begin             low:=ppufile.getint64;             high:=ppufile.getint64;           end         else           begin             low:=ppufile.getlongint;             high:=ppufile.getlongint;           end;         setsize;      end;    function torddef.getcopy : tstoreddef;      begin         result:=torddef.create(typ,low,high);         result.deftype:=orddef;         torddef(result).low:=low;         torddef(result).high:=high;         torddef(result).typ:=typ;         torddef(result).savesize:=savesize;      end;    procedure torddef.setsize;      const        sizetbl : array[tbasetype] of longint = (          0,          1,2,4,8,          1,2,4,8,          1,2,4,          1,2,8        );      begin        savesize:=sizetbl[typ];      end;    function torddef.getvartype : longint;      const        basetype2vartype : array[tbasetype] of longint = (          varUndefined,          varbyte,varqword,varlongword,varqword,          varshortint,varsmallint,varinteger,varint64,          varboolean,varUndefined,varUndefined,          varUndefined,varUndefined,varCurrency);      begin        result:=basetype2vartype[typ];      end;    procedure torddef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.putbyte(byte(typ));         if sizeof(TConstExprInt)=8 then          begin            ppufile.putint64(low);            ppufile.putint64(high);          end         else          begin            ppufile.putlongint(low);            ppufile.putlongint(high);          end;         ppufile.writeentry(iborddef);      end;{$ifdef GDB}    function torddef.stabstring : pchar;      begin        if cs_gdb_valgrind in aktglobalswitches then          begin            case typ of              uvoid :                stabstring := strpnew(numberstring);              bool8bit,              bool16bit,              bool32bit :                stabstring := stabstr_evaluate('r${numberstring};0;255;',[]);              u32bit,              s64bit,              u64bit :                stabstring:=stabstr_evaluate('r${numberstring};0;-1;',[]);              else                stabstring:=stabstr_evaluate('r${numberstring};$1;$2;',[tostr(longint(low)),tostr(longint(high))]);            end;          end        else          begin            case typ of              uvoid :                stabstring := strpnew(numberstring);              uchar :                stabstring := strpnew('-20;');              uwidechar :                stabstring := strpnew('-30;');              bool8bit :                stabstring := strpnew('-21;');              bool16bit :                stabstring := strpnew('-22;');              bool32bit :                stabstring := strpnew('-23;');              u64bit :                stabstring := strpnew('-32;');              s64bit :                stabstring := strpnew('-31;');              {u32bit : stabstring := tstoreddef(s32inttype.def).numberstring+';0;-1;'); }              else                stabstring:=stabstr_evaluate('r${numberstring};$1;$2;',[tostr(longint(low)),tostr(longint(high))]);            end;         end;      end;{$endif GDB}    procedure torddef.write_rtti_data(rt:trttitype);        procedure dointeger;        const          trans : array[tbasetype] of byte =            (otUByte{otNone},             otUByte,otUWord,otULong,otUByte{otNone},             otSByte,otSWord,otSLong,otUByte{otNone},             otUByte,otUWord,otULong,             otUByte,otUWord,otUByte);        begin          write_rtti_name;{$ifdef cpurequiresproperalignment}          rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}          rttiList.concat(Tai_const.Create_8bit(byte(trans[typ])));{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(4));{$endif cpurequiresproperalignment}          rttiList.concat(Tai_const.Create_32bit(longint(low)));          rttiList.concat(Tai_const.Create_32bit(longint(high)));        end;      begin        case typ of          s64bit :            begin              rttiList.concat(Tai_const.Create_8bit(tkInt64));              write_rtti_name;{$ifdef cpurequiresproperalignment}              rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}              { low }              rttiList.concat(Tai_const.Create_64bit(int64($80000000) shl 32));              { high }              rttiList.concat(Tai_const.Create_64bit((int64($7fffffff) shl 32) or int64($ffffffff)));            end;          u64bit :            begin              rttiList.concat(Tai_const.Create_8bit(tkQWord));              write_rtti_name;{$ifdef cpurequiresproperalignment}              rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}              { low }              rttiList.concat(Tai_const.Create_64bit(0));              { high }              rttiList.concat(Tai_const.Create_64bit(int64((int64($ffffffff) shl 32) or int64($ffffffff))));            end;          bool8bit:            begin              rttiList.concat(Tai_const.Create_8bit(tkBool));              dointeger;            end;          uchar:            begin              rttiList.concat(Tai_const.Create_8bit(tkChar));              dointeger;            end;          uwidechar:            begin              rttiList.concat(Tai_const.Create_8bit(tkWChar));              dointeger;            end;          else            begin              rttiList.concat(Tai_const.Create_8bit(tkInteger));              dointeger;            end;        end;      end;    function torddef.is_publishable : boolean;      begin         is_publishable:=(typ<>uvoid);      end;    function torddef.gettypename : string;      const        names : array[tbasetype] of string[20] = (          'untyped',          'Byte','Word','DWord','QWord',          'ShortInt','SmallInt','LongInt','Int64',          'Boolean','WordBool','LongBool',          'Char','WideChar','Currency');      begin         gettypename:=names[typ];      end;{****************************************************************************                                TFLOATDEF****************************************************************************}    constructor tfloatdef.create(t : tfloattype);      begin         inherited create;         deftype:=floatdef;         typ:=t;         setsize;      end;    constructor tfloatdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=floatdef;         typ:=tfloattype(ppufile.getbyte);         setsize;      end;    function tfloatdef.getcopy : tstoreddef;      begin         result:=tfloatdef.create(typ);         result.deftype:=floatdef;         tfloatdef(result).savesize:=savesize;      end;    procedure tfloatdef.setsize;      begin         case typ of           s32real : savesize:=4;           s80real : savesize:=10;           s64real,           s64currency,           s64comp : savesize:=8;         else           savesize:=0;         end;      end;    function tfloatdef.getvartype : longint;      const        floattype2vartype : array[tfloattype] of longint = (          varSingle,varDouble,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:=floattype2vartype[typ];      end;    procedure tfloatdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.putbyte(byte(typ));         ppufile.writeentry(ibfloatdef);      end;{$ifdef GDB}    function Tfloatdef.stabstring:Pchar;      begin        case typ of          s32real,s64real,s80real:            stabstring:=stabstr_evaluate('r$1;${savesize};0;',[tstoreddef(s32inttype.def).numberstring]);          s64currency,s64comp:            stabstring:=stabstr_evaluate('r$1;-${savesize};0;',[tstoreddef(s32inttype.def).numberstring]);          else            internalerror(10005);        end;      end;    procedure tfloatdef.concatstabto(asmlist:taasmoutput);      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;        tstoreddef(s32inttype.def).concatstabto(asmlist);        inherited concatstabto(asmlist);      end;{$endif GDB}    procedure tfloatdef.write_rtti_data(rt:trttitype);      const         {tfloattype = (s32real,s64real,s80real,s64bit,s128bit);}         translate : array[tfloattype] of byte =           (ftSingle,ftDouble,ftExtended,ftComp,ftCurr,ftFloat128);      begin         rttiList.concat(Tai_const.Create_8bit(tkFloat));         write_rtti_name;{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}         rttiList.concat(Tai_const.Create_8bit(translate[typ]));      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','Comp','Currency','Float128');      begin         gettypename:=names[typ];      end;{****************************************************************************                                TFILEDEF****************************************************************************}    constructor tfiledef.createtext;      begin         inherited create;         deftype:=filedef;         filetyp:=ft_text;         typedfiletype.reset;         setsize;      end;    constructor tfiledef.createuntyped;      begin         inherited create;         deftype:=filedef;         filetyp:=ft_untyped;         typedfiletype.reset;         setsize;      end;    constructor tfiledef.createtyped(const tt : ttype);      begin         inherited create;         deftype:=filedef;         filetyp:=ft_typed;         typedfiletype:=tt;         setsize;      end;    constructor tfiledef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=filedef;         filetyp:=tfiletyp(ppufile.getbyte);         if filetyp=ft_typed then           ppufile.gettype(typedfiletype)         else           typedfiletype.reset;         setsize;      end;    function tfiledef.getcopy : tstoreddef;      begin        case filetyp of          ft_typed:            result:=tfiledef.createtyped(typedfiletype);          ft_untyped:            result:=tfiledef.createuntyped;          ft_text:            result:=tfiledef.createtext;          else            internalerror(2004121201);        end;      end;    procedure tfiledef.buildderef;      begin        inherited buildderef;        if filetyp=ft_typed then          typedfiletype.buildderef;      end;    procedure tfiledef.deref;      begin        inherited deref;        if filetyp=ft_typed then          typedfiletype.resolve;      end;    procedure tfiledef.setsize;      begin{$ifdef cpu64bit}        case filetyp of          ft_text :            savesize:=628;          ft_typed,          ft_untyped :            savesize:=368;        end;{$else cpu64bit}        case filetyp of          ft_text :            savesize:=592;          ft_typed,          ft_untyped :            savesize:=332;        end;{$endif cpu64bit}      end;    procedure tfiledef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.putbyte(byte(filetyp));         if filetyp=ft_typed then           ppufile.puttype(typedfiletype);         ppufile.writeentry(ibfiledef);      end;{$ifdef GDB}    function tfiledef.stabstring : pchar;      begin   {$IfDef GDBknowsfiles}      case filetyp of        ft_typed :          stabstring := strpnew('d'+typedfiletype.def.numberstring{+';'});        ft_untyped :          stabstring := strpnew('d'+voiddef.numberstring{+';'});        ft_text :          stabstring := strpnew('d'+cchartype^.numberstring{+';'});      end;   {$Else}{$ifdef cpu64bit}      stabstring:=stabstr_evaluate('s${savesize}HANDLE:$1,0,32;MODE:$1,32,32;RECSIZE:$2,64,64;'+                                   '_PRIVATE:ar$1;1;64;$3,128,256;USERDATA:ar$1;1;16;$3,384,128;'+                                   'NAME:ar$1;0;255;$4,512,2048;;',[tstoreddef(s32inttype.def).numberstring,                                   tstoreddef(s64inttype.def).numberstring,                                   tstoreddef(u8inttype.def).numberstring,                                   tstoreddef(cchartype.def).numberstring]);{$else cpu64bit}      stabstring:=stabstr_evaluate('s${savesize}HANDLE:$1,0,32;MODE:$1,32,32;RECSIZE:$1,64,32;'+                                   '_PRIVATE:ar$1;1;32;$3,96,256;USERDATA:ar$1;1;16;$2,352,128;'+                                   'NAME:ar$1;0;255;$3,480,2048;;',[tstoreddef(s32inttype.def).numberstring,                                   tstoreddef(u8inttype.def).numberstring,                                   tstoreddef(cchartype.def).numberstring]);{$endif cpu64bit}   {$EndIf}      end;    procedure tfiledef.concatstabto(asmlist:taasmoutput);      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;  {$IfDef GDBknowsfiles}        case filetyp of          ft_typed :            tstoreddef(typedfiletype.def).concatstabto(asmlist);          ft_untyped :            tstoreddef(voidtype.def).concatstabto(asmlist);          ft_text :            tstoreddef(cchartype.def).concatstabto(asmlist);        end;  {$Else}        tstoreddef(s32inttype.def).concatstabto(asmlist);{$ifdef cpu64bit}        tstoreddef(s64inttype.def).concatstabto(asmlist);{$endif cpu64bit}        tstoreddef(u8inttype.def).concatstabto(asmlist);        tstoreddef(cchartype.def).concatstabto(asmlist);  {$EndIf}        inherited concatstabto(asmlist);      end;{$endif GDB}    function tfiledef.gettypename : string;      begin         case filetyp of           ft_untyped:             gettypename:='File';           ft_typed:             gettypename:='File Of '+typedfiletype.def.typename;           ft_text:             gettypename:='Text'         end;      end;    function tfiledef.getmangledparaname : string;      begin         case filetyp of           ft_untyped:             getmangledparaname:='FILE';           ft_typed:             getmangledparaname:='FILE$OF$'+typedfiletype.def.mangledparaname;           ft_text:             getmangledparaname:='TEXT'         end;      end;{****************************************************************************                               TVARIANTDEF****************************************************************************}    constructor tvariantdef.create(v : tvarianttype);      begin         inherited create;         varianttype:=v;         deftype:=variantdef;         setsize;      end;    constructor tvariantdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         varianttype:=tvarianttype(ppufile.getbyte);         deftype:=variantdef;         setsize;      end;    function tvariantdef.getcopy : tstoreddef;      begin        result:=tvariantdef.create(varianttype);      end;    procedure tvariantdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.putbyte(byte(varianttype));         ppufile.writeentry(ibvariantdef);      end;    procedure tvariantdef.setsize;      begin         savesize:=16;      end;    function tvariantdef.gettypename : string;      begin         case varianttype of           vt_normalvariant:             gettypename:='Variant';           vt_olevariant:             gettypename:='OleVariant';         end;      end;    procedure tvariantdef.write_rtti_data(rt:trttitype);      begin         rttiList.concat(Tai_const.Create_8bit(tkVariant));      end;    function tvariantdef.needs_inittable : boolean;      begin         needs_inittable:=true;      end;{$ifdef GDB}    function tvariantdef.stabstring : pchar;      begin        stabstring:=stabstr_evaluate('formal${numberstring};',[]);      end;    function tvariantdef.numberstring:string;      begin        result:=tstoreddef(voidtype.def).numberstring;      end;    procedure tvariantdef.concatstabto(asmlist : taasmoutput);      begin        { don't know how to handle this }      end;{$endif GDB}    function tvariantdef.is_publishable : boolean;      begin         is_publishable:=true;      end;{****************************************************************************                               TPOINTERDEF****************************************************************************}    constructor tpointerdef.create(const tt : ttype);      begin        inherited create;        deftype:=pointerdef;        pointertype:=tt;        is_far:=false;        savesize:=sizeof(aint);      end;    constructor tpointerdef.createfar(const tt : ttype);      begin        inherited create;        deftype:=pointerdef;        pointertype:=tt;        is_far:=true;        savesize:=sizeof(aint);      end;    constructor tpointerdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=pointerdef;         ppufile.gettype(pointertype);         is_far:=(ppufile.getbyte<>0);         savesize:=sizeof(aint);      end;    function tpointerdef.getcopy : tstoreddef;      begin        result:=tpointerdef.create(pointertype);        tpointerdef(result).is_far:=is_far;        tpointerdef(result).savesize:=savesize;      end;    procedure tpointerdef.buildderef;      begin        inherited buildderef;        pointertype.buildderef;      end;    procedure tpointerdef.deref;      begin        inherited deref;        pointertype.resolve;      end;    procedure tpointerdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.puttype(pointertype);         ppufile.putbyte(byte(is_far));         ppufile.writeentry(ibpointerdef);      end;{$ifdef GDB}    function tpointerdef.stabstring : pchar;      begin        stabstring := strpnew('*'+tstoreddef(pointertype.def).numberstring);      end;    procedure tpointerdef.concatstabto(asmlist : taasmoutput);      var st,nb : string;      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;        stab_state:=stab_state_writing;        tstoreddef(pointertype.def).concatstabto(asmlist);        if (pointertype.def.deftype in [recorddef,objectdef]) then          begin            if pointertype.def.deftype=objectdef then              nb:=tobjectdef(pointertype.def).classnumberstring            else              nb:=tstoreddef(pointertype.def).numberstring;            {to avoid infinite recursion in record with next-like fields }            if tstoreddef(pointertype.def).stab_state=stab_state_writing then              begin                if assigned(pointertype.def.typesym) then                  begin                    if assigned(typesym) then                      st := ttypesym(typesym).name                    else                      st := ' ';                    asmlist.concat(Tai_stabs.create(stabstr_evaluate(                            '"$1:t${numberstring}=*$2=xs$3:",${N_LSYM},0,0,0',                            [st,nb,pointertype.def.typesym.name])));                  end;                stab_state:=stab_state_written;              end            else              begin                stab_state:=stab_state_used;                inherited concatstabto(asmlist);              end;          end        else          begin            stab_state:=stab_state_used;            inherited concatstabto(asmlist);          end;      end;{$endif GDB}    function tpointerdef.gettypename : string;      begin         if is_far then          gettypename:='^'+pointertype.def.typename+';far'         else          gettypename:='^'+pointertype.def.typename;      end;{****************************************************************************                              TCLASSREFDEF****************************************************************************}    constructor tclassrefdef.create(const t:ttype);      begin         inherited create(t);         deftype:=classrefdef;      end;    constructor tclassrefdef.ppuload(ppufile:tcompilerppufile);      begin         { be careful, tclassdefref inherits from tpointerdef }         inherited ppuloaddef(ppufile);         deftype:=classrefdef;         ppufile.gettype(pointertype);         is_far:=false;         savesize:=sizeof(aint);      end;    procedure tclassrefdef.ppuwrite(ppufile:tcompilerppufile);      begin         { be careful, tclassdefref inherits from tpointerdef }         inherited ppuwritedef(ppufile);         ppufile.puttype(pointertype);         ppufile.writeentry(ibclassrefdef);      end;{$ifdef GDB}    function tclassrefdef.stabstring : pchar;      begin         stabstring:=strpnew(tstoreddef(pvmttype.def).numberstring);      end;{$endif GDB}    function tclassrefdef.gettypename : string;      begin         gettypename:='Class Of '+pointertype.def.typename;      end;    function tclassrefdef.is_publishable : boolean;      begin         is_publishable:=true;      end;{***************************************************************************                                   TSETDEF***************************************************************************}    constructor tsetdef.create(const t:ttype;high : aint);      begin         inherited create;         deftype:=setdef;         elementtype:=t;         // setbase:=low;         setmax:=high;         if high<32 then           begin            settype:=smallset;           {$ifdef testvarsets}            if aktsetalloc=0 THEN      { $PACKSET Fixed?}           {$endif}            savesize:=Sizeof(longint)           {$ifdef testvarsets}           else                       {No, use $PACKSET VALUE for rounding}            savesize:=aktsetalloc*((high+aktsetalloc*8-1) DIV (aktsetalloc*8))           {$endif}              ;          end         else          if high<256 then           begin              settype:=normset;              savesize:=32;           end         else{$ifdef testvarsets}         if high<$10000 then           begin              settype:=varset;              savesize:=4*((high+31) div 32);           end         else{$endif testvarsets}          Message(sym_e_ill_type_decl_set);      end;    constructor tsetdef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=setdef;         ppufile.gettype(elementtype);         settype:=tsettype(ppufile.getbyte);         case settype of           normset : savesize:=32;           varset : savesize:=ppufile.getlongint;           smallset : savesize:=Sizeof(longint);         end;      end;    destructor tsetdef.destroy;      begin        inherited destroy;      end;    function tsetdef.getcopy : tstoreddef;      begin        case settype of          smallset:            result:=tsetdef.create(elementtype,31);          normset:            result:=tsetdef.create(elementtype,255);          else            internalerror(2004121202);        end;      end;    procedure tsetdef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.puttype(elementtype);         ppufile.putbyte(byte(settype));         if settype=varset then           ppufile.putlongint(savesize);         if settype=normset then           ppufile.putaint(savesize);         ppufile.writeentry(ibsetdef);      end;{$ifdef GDB}    function tsetdef.stabstring : pchar;      begin        stabstring:=stabstr_evaluate('@s$1;S$2',[tostr(savesize*8),tstoreddef(elementtype.def).numberstring]);      end;    procedure tsetdef.concatstabto(asmlist:taasmoutput);      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;        tstoreddef(elementtype.def).concatstabto(asmlist);        inherited concatstabto(asmlist);      end;{$endif GDB}    procedure tsetdef.buildderef;      begin        inherited buildderef;        elementtype.buildderef;      end;    procedure tsetdef.deref;      begin        inherited deref;        elementtype.resolve;      end;    procedure tsetdef.write_child_rtti_data(rt:trttitype);      begin        tstoreddef(elementtype.def).get_rtti_label(rt);      end;    procedure tsetdef.write_rtti_data(rt:trttitype);      begin         rttiList.concat(Tai_const.Create_8bit(tkSet));         write_rtti_name;{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}         rttiList.concat(Tai_const.Create_8bit(otULong));{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}         rttiList.concat(Tai_const.Create_sym(tstoreddef(elementtype.def).get_rtti_label(rt)));      end;    function tsetdef.is_publishable : boolean;      begin         is_publishable:=(settype=smallset);      end;    function tsetdef.gettypename : string;      begin         if assigned(elementtype.def) then          gettypename:='Set Of '+elementtype.def.typename         else          gettypename:='Empty Set';      end;{***************************************************************************                                 TFORMALDEF***************************************************************************}    constructor tformaldef.create;      var         stregdef : boolean;      begin         stregdef:=registerdef;         registerdef:=false;         inherited create;         deftype:=formaldef;         registerdef:=stregdef;         { formaldef must be registered at unit level !! }         if registerdef and assigned(current_module) then            if assigned(current_module.localsymtable) then              tsymtable(current_module.localsymtable).registerdef(self)            else if assigned(current_module.globalsymtable) then              tsymtable(current_module.globalsymtable).registerdef(self);         savesize:=0;      end;    constructor tformaldef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=formaldef;         savesize:=0;      end;    procedure tformaldef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.writeentry(ibformaldef);      end;{$ifdef GDB}    function tformaldef.stabstring : pchar;      begin        stabstring:=stabstr_evaluate('formal${numberstring};',[]);      end;    function tformaldef.numberstring:string;      begin        result:=tstoreddef(voidtype.def).numberstring;      end;    procedure tformaldef.concatstabto(asmlist : taasmoutput);      begin        { formaldef can't be stab'ed !}      end;{$endif GDB}    function tformaldef.gettypename : string;      begin         gettypename:='<Formal type>';      end;{***************************************************************************                           TARRAYDEF***************************************************************************}    constructor tarraydef.create(l,h : aint;const t : ttype);      begin         inherited create;         deftype:=arraydef;         lowrange:=l;         highrange:=h;         rangetype:=t;         elementtype.reset;         IsVariant:=false;         IsConstructor:=false;         IsArrayOfConst:=false;         IsDynamicArray:=false;         IsConvertedPointer:=false;      end;    constructor tarraydef.create_from_pointer(const elemt : ttype);      begin         self.create(0,$7fffffff,s32inttype);         IsConvertedPointer:=true;         setelementtype(elemt);      end;    constructor tarraydef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=arraydef;         { the addresses are calculated later }         ppufile.gettype(_elementtype);         ppufile.gettype(rangetype);         lowrange:=ppufile.getaint;         highrange:=ppufile.getaint;         IsArrayOfConst:=boolean(ppufile.getbyte);         IsDynamicArray:=boolean(ppufile.getbyte);         IsVariant:=false;         IsConstructor:=false;      end;    function tarraydef.getcopy : tstoreddef;      begin        result:=tarraydef.create(lowrange,highrange,rangetype);        tarraydef(result).IsConvertedPointer:=IsConvertedPointer;        tarraydef(result).IsDynamicArray:=IsDynamicArray;        tarraydef(result).IsVariant:=IsVariant;        tarraydef(result).IsConstructor:=IsConstructor;        tarraydef(result).IsArrayOfConst:=IsArrayOfConst;        tarraydef(result)._elementtype:=_elementtype;      end;    procedure tarraydef.buildderef;      begin        inherited buildderef;        _elementtype.buildderef;        rangetype.buildderef;      end;    procedure tarraydef.deref;      begin        inherited deref;        _elementtype.resolve;        rangetype.resolve;      end;    procedure tarraydef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.puttype(_elementtype);         ppufile.puttype(rangetype);         ppufile.putaint(lowrange);         ppufile.putaint(highrange);         ppufile.putbyte(byte(IsArrayOfConst));         ppufile.putbyte(byte(IsDynamicArray));         ppufile.writeentry(ibarraydef);      end;{$ifdef GDB}    function tarraydef.stabstring : pchar;      begin        stabstring:=stabstr_evaluate('ar$1;$2;$3;$4',[Tstoreddef(rangetype.def).numberstring,                    tostr(lowrange),tostr(highrange),Tstoreddef(_elementtype.def).numberstring]);      end;    procedure tarraydef.concatstabto(asmlist:taasmoutput);      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;        tstoreddef(rangetype.def).concatstabto(asmlist);        tstoreddef(_elementtype.def).concatstabto(asmlist);        inherited concatstabto(asmlist);      end;{$endif GDB}    function tarraydef.elesize : aint;      begin        elesize:=_elementtype.def.size;      end;    function tarraydef.elecount : aint;      var        qhigh,qlow : qword;      begin        if IsDynamicArray then          begin            result:=0;            exit;          end;        if (highrange>0) and (lowrange<0) then          begin            qhigh:=highrange;            qlow:=qword(-lowrange);            { prevent overflow, return -1 to indicate overflow }            if qhigh+qlow>qword(high(aint)-1) then              result:=-1            else              result:=qhigh+qlow+1;          end        else          result:=int64(highrange)-lowrange+1;      end;    function tarraydef.size : aint;      var        cachedelecount,        cachedelesize : aint;      begin        if IsDynamicArray then          begin            size:=sizeof(aint);            exit;          end;        { Tarraydef.size may never be called for an open array! }        if highrange<lowrange then          internalerror(99080501);        cachedelesize:=elesize;        cachedelecount:=elecount;        { prevent overflow, return -1 to indicate overflow }        if (cachedelesize <> 0) and           (            (cachedelecount < 0) or            ((high(aint) div cachedelesize) < cachedelecount) or            { also lowrange*elesize must be < high(aint) to prevent overflow when              accessing the array, see ncgmem (PFV) }            ((high(aint) div cachedelesize) < abs(lowrange))           ) then          result:=-1        else          result:=cachedelesize*cachedelecount;      end;    procedure tarraydef.setelementtype(t: ttype);      begin        _elementtype:=t;       if not(IsDynamicArray or              IsConvertedPointer or              (highrange<lowrange)) then         begin           if (size=-1) then             Message(sym_e_segment_too_large);         end;      end;    function tarraydef.alignment : longint;      begin         { alignment is the size of the elements }         if (elementtype.def.deftype in [arraydef,recorddef]) or           ((elementtype.def.deftype=objectdef) and             is_object(elementtype.def)) then           alignment:=elementtype.def.alignment         else           alignment:=elesize;      end;    function tarraydef.needs_inittable : boolean;      begin         needs_inittable:=IsDynamicArray or elementtype.def.needs_inittable;      end;    procedure tarraydef.write_child_rtti_data(rt:trttitype);      begin        tstoreddef(elementtype.def).get_rtti_label(rt);      end;    procedure tarraydef.write_rtti_data(rt:trttitype);      begin         if IsDynamicArray then           rttiList.concat(Tai_const.Create_8bit(tkdynarray))         else           rttiList.concat(Tai_const.Create_8bit(tkarray));         write_rtti_name;{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}         { size of elements }         rttiList.concat(Tai_const.Create_aint(elesize));         if not(IsDynamicArray) then           rttiList.concat(Tai_const.Create_aint(elecount));         { element type }         rttiList.concat(Tai_const.Create_sym(tstoreddef(elementtype.def).get_rtti_label(rt)));         { variant type }         rttilist.concat(Tai_const.Create_32bit(tstoreddef(elementtype.def).getvartype));      end;    function tarraydef.gettypename : string;      begin         if isarrayofconst or isConstructor then           begin             if isvariant or ((highrange=-1) and (lowrange=0)) then               gettypename:='Array Of Const'             else               gettypename:='Array Of '+elementtype.def.typename;           end         else if ((highrange=-1) and (lowrange=0)) or IsDynamicArray then           gettypename:='Array Of '+elementtype.def.typename         else           begin              if rangetype.def.deftype=enumdef then                gettypename:='Array['+rangetype.def.typename+'] Of '+elementtype.def.typename              else                gettypename:='Array['+tostr(lowrange)+'..'+                  tostr(highrange)+'] Of '+elementtype.def.typename           end;      end;    function tarraydef.getmangledparaname : string;      begin         if isarrayofconst then          getmangledparaname:='array_of_const'         else          if ((highrange=-1) and (lowrange=0)) then           getmangledparaname:='array_of_'+elementtype.def.mangledparaname         else          internalerror(200204176);      end;{***************************************************************************                              tabstractrecorddef***************************************************************************}    function tabstractrecorddef.getsymtable(t:tgetsymtable):tsymtable;      begin         if t=gs_record then         getsymtable:=symtable        else         getsymtable:=nil;      end;{$ifdef GDB}    procedure tabstractrecorddef.field_addname(p:Tnamedindexitem;arg:pointer);      var        newrec:Pchar;        spec:string[3];        varsize : aint;        state   : ^Trecord_stabgen_state;      begin        state:=arg;        { static variables from objects are like global objects }        if (Tsym(p).typ=fieldvarsym) and not (sp_static in Tsym(p).symoptions) then          begin            if ([sp_protected,sp_strictprotected]*tsym(p).symoptions)<>[] then              spec:='/1'            else if ([sp_private,sp_strictprivate]*tsym(p).symoptions)<>[] then              spec:='/0'            else              spec:='';            varsize:=tfieldvarsym(p).vartype.def.size;            { open arrays made overflows !! }            if varsize>$fffffff then              varsize:=$fffffff;            newrec:=stabstr_evaluate('$1:$2,$3,$4;',[p.name,                                     spec+tstoreddef(tfieldvarsym(p).vartype.def).numberstring,                                     tostr(tfieldvarsym(p).fieldoffset*8),tostr(varsize*8)]);            if state^.stabsize+strlen(newrec)>=state^.staballoc-256 then              begin                inc(state^.staballoc,memsizeinc);                reallocmem(state^.stabstring,state^.staballoc);              end;            strcopy(state^.stabstring+state^.stabsize,newrec);            inc(state^.stabsize,strlen(newrec));            strdispose(newrec);            {This should be used for case !!}            inc(state^.recoffset,Tfieldvarsym(p).vartype.def.size);          end;      end;    procedure tabstractrecorddef.field_concatstabto(p:Tnamedindexitem;arg:pointer);      begin        if (Tsym(p).typ=fieldvarsym) and not (sp_static in Tsym(p).symoptions) then          tstoreddef(tfieldvarsym(p).vartype.def).concatstabto(taasmoutput(arg));      end;{$endif GDB}    procedure tabstractrecorddef.count_field_rtti(sym : tnamedindexitem;arg:pointer);      begin         if (FRTTIType=fullrtti) or            ((tsym(sym).typ=fieldvarsym) and             tfieldvarsym(sym).vartype.def.needs_inittable) then           inc(Count);      end;    procedure tabstractrecorddef.generate_field_rtti(sym:tnamedindexitem;arg:pointer);      begin         if (FRTTIType=fullrtti) or            ((tsym(sym).typ=fieldvarsym) and             tfieldvarsym(sym).vartype.def.needs_inittable) then           tstoreddef(tfieldvarsym(sym).vartype.def).get_rtti_label(FRTTIType);      end;    procedure tabstractrecorddef.write_field_rtti(sym : tnamedindexitem;arg:pointer);      begin         if (FRTTIType=fullrtti) or            ((tsym(sym).typ=fieldvarsym) and             tfieldvarsym(sym).vartype.def.needs_inittable) then          begin            rttiList.concat(Tai_const.Create_sym(tstoreddef(tfieldvarsym(sym).vartype.def).get_rtti_label(FRTTIType)));            rttiList.concat(Tai_const.Create_32bit(tfieldvarsym(sym).fieldoffset));          end;      end;{***************************************************************************                                  trecorddef***************************************************************************}    constructor trecorddef.create(p : tsymtable);      begin         inherited create;         deftype:=recorddef;         symtable:=p;         symtable.defowner:=self;         isunion:=false;      end;    constructor trecorddef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuloaddef(ppufile);         deftype:=recorddef;         symtable:=trecordsymtable.create(0);         trecordsymtable(symtable).datasize:=ppufile.getaint;         trecordsymtable(symtable).fieldalignment:=shortint(ppufile.getbyte);         trecordsymtable(symtable).recordalignment:=shortint(ppufile.getbyte);         trecordsymtable(symtable).padalignment:=shortint(ppufile.getbyte);         trecordsymtable(symtable).ppuload(ppufile);         symtable.defowner:=self;         isunion:=false;      end;    destructor trecorddef.destroy;      begin         if assigned(symtable) then           symtable.free;         inherited destroy;      end;    function trecorddef.getcopy : tstoreddef;      begin        result:=trecorddef.create(symtable.getcopy);        trecorddef(result).isunion:=isunion;      end;    function trecorddef.needs_inittable : boolean;      begin        needs_inittable:=trecordsymtable(symtable).needs_init_final      end;    procedure trecorddef.buildderef;      var         oldrecsyms : tsymtable;      begin         inherited buildderef;         oldrecsyms:=aktrecordsymtable;         aktrecordsymtable:=symtable;         { now build the definitions }         tstoredsymtable(symtable).buildderef;         aktrecordsymtable:=oldrecsyms;      end;    procedure trecorddef.deref;      var         oldrecsyms : tsymtable;      begin         inherited deref;         oldrecsyms:=aktrecordsymtable;         aktrecordsymtable:=symtable;         { now dereference the definitions }         tstoredsymtable(symtable).deref;         aktrecordsymtable:=oldrecsyms;         { assign TGUID? load only from system unit }         if not(assigned(rec_tguid)) and            (upper(typename)='TGUID') and            assigned(owner) and            assigned(owner.name) and            (owner.name^='SYSTEM') then           rec_tguid:=self;      end;    procedure trecorddef.ppuwrite(ppufile:tcompilerppufile);      begin         inherited ppuwritedef(ppufile);         ppufile.putaint(trecordsymtable(symtable).datasize);         ppufile.putbyte(byte(trecordsymtable(symtable).fieldalignment));         ppufile.putbyte(byte(trecordsymtable(symtable).recordalignment));         ppufile.putbyte(byte(trecordsymtable(symtable).padalignment));         ppufile.writeentry(ibrecorddef);         trecordsymtable(symtable).ppuwrite(ppufile);      end;    function trecorddef.size:aint;      begin        result:=trecordsymtable(symtable).datasize;      end;    function trecorddef.alignment:longint;      begin        alignment:=trecordsymtable(symtable).recordalignment;      end;    function trecorddef.padalignment:longint;      begin        padalignment := trecordsymtable(symtable).padalignment;      end;{$ifdef GDB}    function trecorddef.stabstring : pchar;      var        state:Trecord_stabgen_state;      begin        getmem(state.stabstring,memsizeinc);        state.staballoc:=memsizeinc;        strpcopy(state.stabstring,'s'+tostr(size));        state.recoffset:=0;        state.stabsize:=strlen(state.stabstring);        symtable.foreach(@field_addname,@state);        state.stabstring[state.stabsize]:=';';        state.stabstring[state.stabsize+1]:=#0;        reallocmem(state.stabstring,state.stabsize+2);        stabstring:=state.stabstring;      end;    procedure trecorddef.concatstabto(asmlist:taasmoutput);      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;        symtable.foreach(@field_concatstabto,asmlist);        inherited concatstabto(asmlist);      end;{$endif GDB}    procedure trecorddef.write_child_rtti_data(rt:trttitype);      begin         FRTTIType:=rt;         symtable.foreach(@generate_field_rtti,nil);      end;    procedure trecorddef.write_rtti_data(rt:trttitype);      begin         rttiList.concat(Tai_const.Create_8bit(tkrecord));         write_rtti_name;{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}         rttiList.concat(Tai_const.Create_32bit(size));         Count:=0;         FRTTIType:=rt;         symtable.foreach(@count_field_rtti,nil);         rttiList.concat(Tai_const.Create_32bit(Count));         symtable.foreach(@write_field_rtti,nil);      end;    function trecorddef.gettypename : string;      begin         gettypename:='<record type>'      end;{***************************************************************************                       TABSTRACTPROCDEF***************************************************************************}    constructor tabstractprocdef.create(level:byte);      begin         inherited create;         parast:=tparasymtable.create(level);         parast.defowner:=self;         parast.next:=owner;         paras:=nil;         minparacount:=0;         maxparacount:=0;         proctypeoption:=potype_none;         proccalloption:=pocall_none;         procoptions:=[];         rettype:=voidtype;{$ifdef i386}         fpu_used:=0;{$endif i386}         savesize:=sizeof(aint);         requiredargarea:=0;         has_paraloc_info:=false;         location_reset(funcretloc[callerside],LOC_INVALID,OS_NO);         location_reset(funcretloc[calleeside],LOC_INVALID,OS_NO);      end;    destructor tabstractprocdef.destroy;      begin         if assigned(paras) then           begin{$ifdef MEMDEBUG}             memprocpara.start;{$endif MEMDEBUG}             paras.free;{$ifdef MEMDEBUG}             memprocpara.stop;{$endif MEMDEBUG}          end;         if assigned(parast) then          begin{$ifdef MEMDEBUG}            memprocparast.start;{$endif MEMDEBUG}            parast.free;{$ifdef MEMDEBUG}            memprocparast.stop;{$endif MEMDEBUG}          end;         inherited destroy;      end;    procedure tabstractprocdef.releasemem;      begin        if assigned(paras) then          begin            paras.free;            paras:=nil;          end;        parast.free;        parast:=nil;      end;    procedure tabstractprocdef.count_para(p:tnamedindexitem;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:tnamedindexitem;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;        paracount:=0;        minparacount:=0;        maxparacount:=0;        parast.foreach(@count_para,@paracount);        paras.capacity:=paracount;        { Insert parameters in table }        parast.foreach(@insert_para,nil);        { Order parameters }        paras.sortparas;      end;    { all functions returning in FPU are      assume to use 2 FPU registers      until the function implementation      is processed   PM }    procedure tabstractprocdef.test_if_fpu_result;      begin{$ifdef i386}         if assigned(rettype.def) and            (rettype.def.deftype=floatdef) then           fpu_used:=maxfpuregs;{$endif i386}      end;    procedure tabstractprocdef.buildderef;      begin         { released procdef? }         if not assigned(parast) then           exit;         inherited buildderef;         rettype.buildderef;         { parast }         tparasymtable(parast).buildderef;      end;    procedure tabstractprocdef.deref;      begin         inherited deref;         rettype.resolve;         { parast }         tparasymtable(parast).deref;         { recalculated parameters }         calcparas;      end;    constructor tabstractprocdef.ppuload(ppufile:tcompilerppufile);      var        b : byte;      begin         inherited ppuloaddef(ppufile);         parast:=nil;         Paras:=nil;         minparacount:=0;         maxparacount:=0;         ppufile.gettype(rettype);{$ifdef i386}         fpu_used:=ppufile.getbyte;{$else}         ppufile.getbyte;{$endif i386}         proctypeoption:=tproctypeoption(ppufile.getbyte);         proccalloption:=tproccalloption(ppufile.getbyte);         ppufile.getnormalset(procoptions);         location_reset(funcretloc[callerside],LOC_INVALID,OS_NO);         location_reset(funcretloc[calleeside],LOC_INVALID,OS_NO);         if po_explicitparaloc in procoptions then           begin             b:=ppufile.getbyte;             if b<>sizeof(funcretloc[callerside]) then               internalerror(200411154);             ppufile.getdata(funcretloc[callerside],sizeof(funcretloc[callerside]));           end;         savesize:=sizeof(aint);         has_paraloc_info:=(po_explicitparaloc in procoptions);      end;    procedure tabstractprocdef.ppuwrite(ppufile:tcompilerppufile);      var        oldintfcrc : boolean;      begin         { released procdef? }         if not assigned(parast) then           exit;         inherited ppuwritedef(ppufile);         ppufile.puttype(rettype);         oldintfcrc:=ppufile.do_interface_crc;         ppufile.do_interface_crc:=false;{$ifdef i386}         if simplify_ppu then          fpu_used:=0;         ppufile.putbyte(fpu_used);{$else}         ppufile.putbyte(0);{$endif}         ppufile.putbyte(ord(proctypeoption));         ppufile.putbyte(ord(proccalloption));         ppufile.putnormalset(procoptions);         ppufile.do_interface_crc:=oldintfcrc;         if (po_explicitparaloc in procoptions) then           begin             { Make a 'valid' funcretloc for procedures }             ppufile.putbyte(sizeof(funcretloc[callerside]));             ppufile.putdata(funcretloc[callerside],sizeof(funcretloc[callerside]));           end;      end;    function tabstractprocdef.typename_paras(showhidden:boolean) : string;      var        hs,s  : string;        hp    : TParavarsym;        hpc   : tconstsym;        first : boolean;        i     : 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              (showhidden) then            begin               if first then                begin                  s:=s+'(';                  first:=false;                end               else                s:=s+',';               case hp.varspez of                 vs_var :                   s:=s+'var';                 vs_const :                   s:=s+'const';                 vs_out :                   s:=s+'out';               end;               if assigned(hp.vartype.def.typesym) then                 begin                   if s<>'(' then                    s:=s+' ';                   hs:=hp.vartype.def.typesym.realname;                   if hs[1]<>'$' then                     s:=s+hp.vartype.def.typesym.realname                   else                     s:=s+hp.vartype.def.gettypename;                 end               else                 s:=s+hp.vartype.def.gettypename;               { default value }               if assigned(hp.defaultconstsym) then                begin                  hpc:=tconstsym(hp.defaultconstsym);                  hs:='';                  case hpc.consttyp of                    conststring,                    constresourcestring :                      hs:=strpas(pchar(hpc.value.valueptr));                    constreal :                      str(pbestreal(hpc.value.valueptr)^,hs);                    constpointer :                      hs:=tostr(hpc.value.valueordptr);                    constord :                      begin                        if is_boolean(hpc.consttype.def) 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>';                  end;                  if hs<>'' then                   s:=s+'="'+hs+'"';                end;             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;{$ifdef GDB}    function tabstractprocdef.stabstring : pchar;      begin        stabstring := strpnew('abstractproc'+numberstring+';');      end;{$endif GDB}{***************************************************************************                                  TPROCDEF***************************************************************************}    constructor tprocdef.create(level:byte);      begin         inherited create(level);         deftype:=procdef;         _mangledname:=nil;         fileinfo:=aktfilepos;         extnumber:=$ffff;         aliasnames:=tstringlist.create;         funcretsym:=nil;         localst := nil;         defref:=nil;         lastwritten:=nil;         refcount:=0;         if (cs_browser in aktmoduleswitches) and make_ref then          begin            defref:=tref.create(defref,@akttokenpos);            inc(refcount);          end;         lastref:=defref;         forwarddef:=true;         interfacedef:=false;         hasforward:=false;         _class := nil;         import_dll:=nil;         import_name:=nil;         import_nr:=0;         inlininginfo:=nil;{$ifdef GDB}         isstabwritten := false;{$endif GDB}      end;    constructor tprocdef.ppuload(ppufile:tcompilerppufile);      var        level : byte;      begin         inherited ppuload(ppufile);         deftype:=procdef;         if po_has_mangledname in procoptions then          _mangledname:=stringdup(ppufile.getstring)         else          _mangledname:=nil;         extnumber:=ppufile.getword;         level:=ppufile.getbyte;         ppufile.getderef(_classderef);         ppufile.getderef(procsymderef);         ppufile.getposinfo(fileinfo);         ppufile.getsmallset(symoptions);{$ifdef powerpc}         { library symbol for AmigaOS/MorphOS }         ppufile.getderef(libsymderef);{$endif powerpc}         { import stuff }         import_dll:=nil;         import_name:=nil;         import_nr:=0;         { inline stuff }         if (po_has_inlininginfo in procoptions) then           begin             ppufile.getderef(funcretsymderef);             new(inlininginfo);             ppufile.getsmallset(inlininginfo^.flags);           end         else           begin             inlininginfo:=nil;             funcretsym:=nil;           end;         { load para symtable }         parast:=tparasymtable.create(level);         tparasymtable(parast).ppuload(ppufile);         parast.defowner:=self;         { load local symtable }         if (po_has_inlininginfo in procoptions) or            ((current_module.flags and uf_local_browser)<>0) then          begin            localst:=tlocalsymtable.create(level);            tlocalsymtable(localst).ppuload(ppufile);            localst.defowner:=self;          end         else          localst:=nil;         { inline stuff }         if (po_has_inlininginfo in procoptions) then           inlininginfo^.code:=ppuloadnodetree(ppufile);         { default values for no persistent data }         if (cs_link_deffile in aktglobalswitches) and            (tf_need_export in target_info.flags) and            (po_exports in procoptions) then           deffile.AddExport(mangledname);         aliasnames:=tstringlist.create;         forwarddef:=false;         interfacedef:=false;         hasforward:=false;         lastref:=nil;         lastwritten:=nil;         defref:=nil;         refcount:=0;{$ifdef GDB}         isstabwritten := false;{$endif GDB}         { Disable po_has_inlining until the derefimpl is done }         exclude(procoptions,po_has_inlininginfo);      end;    destructor tprocdef.destroy;      begin         if assigned(defref) then           begin             defref.freechain;             defref.free;           end;         aliasnames.free;         if assigned(localst) and (localst.symtabletype<>staticsymtable) then          begin{$ifdef MEMDEBUG}            memproclocalst.start;{$endif MEMDEBUG}            localst.free;{$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);          end;         stringdispose(import_dll);         stringdispose(import_name);         if (po_msgstr in procoptions) then           strdispose(messageinf.str);         if assigned(_mangledname) then          begin{$ifdef MEMDEBUG}            memmanglednames.start;{$endif MEMDEBUG}            stringdispose(_mangledname);{$ifdef MEMDEBUG}            memmanglednames.stop;{$endif MEMDEBUG}          end;         inherited destroy;      end;    procedure tprocdef.ppuwrite(ppufile:tcompilerppufile);      var        oldintfcrc : boolean;        oldparasymtable,        oldlocalsymtable : tsymtable;      begin         { released procdef? }         if not assigned(parast) then           exit;         oldparasymtable:=aktparasymtable;         oldlocalsymtable:=aktlocalsymtable;         aktparasymtable:=parast;         aktlocalsymtable:=localst;         inherited ppuwrite(ppufile);         oldintfcrc:=ppufile.do_interface_crc;         ppufile.do_interface_crc:=false;         ppufile.do_interface_crc:=oldintfcrc;         if po_has_mangledname in procoptions then          ppufile.putstring(_mangledname^);         ppufile.putword(extnumber);         ppufile.putbyte(parast.symtablelevel);         ppufile.putderef(_classderef);         ppufile.putderef(procsymderef);         ppufile.putposinfo(fileinfo);         ppufile.putsmallset(symoptions);{$ifdef powerpc}         { library symbol for AmigaOS/MorphOS }         ppufile.putderef(libsymderef);{$endif powerpc}         { inline stuff }         oldintfcrc:=ppufile.do_crc;         ppufile.do_crc:=false;         if (po_has_inlininginfo in procoptions) then           begin             ppufile.putderef(funcretsymderef);             ppufile.putsmallset(inlininginfo^.flags);           end;         ppufile.do_crc:=oldintfcrc;         { write this entry }         ppufile.writeentry(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 (po_has_inlininginfo in procoptions) or            ((current_module.flags and uf_local_browser)<>0) then          begin            { we must write a localsymtable }            if not assigned(localst) then              insert_localst;            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 (po_has_inlininginfo in procoptions) then           ppuwritenodetree(ppufile,inlininginfo^.code);         ppufile.do_crc:=oldintfcrc;         aktparasymtable:=oldparasymtable;         aktlocalsymtable:=oldlocalsymtable;      end;    procedure tprocdef.insert_localst;      begin         localst:=tlocalsymtable.create(parast.symtablelevel);         localst.defowner:=self;         { this is used by insert           to check same names in parast and localst }         localst.next:=parast;      end;    function tprocdef.fullprocname(showhidden:boolean):string;      var        s : string;        t : ttoken;      begin{$ifdef EXTDEBUG}        showhidden:=true;{$endif EXTDEBUG}        s:='';        if owner.symtabletype=localsymtable then         s:=s+'local ';        if assigned(_class) then         begin           if po_classmethod in procoptions then            s:=s+'class ';           s:=s+_class.objrealname^+'.';         end;        if proctypeoption=potype_operator then          begin            for t:=NOTOKEN to last_overloaded do              if procsym.realname='$'+overloaded_names[t] then                begin                  s:='operator '+arraytokeninfo[t].str+typename_paras(showhidden);                  break;                end;          end        else          s:=s+procsym.realname+typename_paras(showhidden);        case proctypeoption of          potype_constructor:            s:='constructor '+s;          potype_destructor:            s:='destructor '+s;          else            if assigned(rettype.def) and              not(is_void(rettype.def)) then              s:=s+':'+rettype.def.gettypename;        end;        { forced calling convention? }        if (po_hascallingconvention in procoptions) then          s:=s+';'+ProcCallOptionStr[proccalloption];        fullprocname:=s;      end;    function tprocdef.is_methodpointer:boolean;      begin        result:=assigned(_class);      end;    function tprocdef.is_addressonly:boolean;      begin        result:=assigned(owner) and                (owner.symtabletype<>objectsymtable);      end;    function tprocdef.is_visible_for_object(currobjdef:tobjectdef):boolean;      begin        is_visible_for_object:=false;        { private symbols are allowed when we are in the same          module as they are defined }        if (sp_private in symoptions) and           (owner.defowner.owner.symtabletype in [globalsymtable,staticsymtable]) and           not(owner.defowner.owner.iscurrentunit) then          exit;        if (sp_strictprivate in symoptions) then          begin            result:=currobjdef=tobjectdef(owner.defowner);            exit;          end;        if (sp_strictprotected in symoptions) then          begin             result:=assigned(currobjdef) and               currobjdef.is_related(tobjectdef(owner.defowner));             exit;          end;        { protected symbols are visible in the module that defines them and          also visible to related objects. The related object must be defined          in the current module }        if (sp_protected in symoptions) and           (            (             (owner.defowner.owner.symtabletype in [globalsymtable,staticsymtable]) and             not(owner.defowner.owner.iscurrentunit)            ) and            not(                assigned(currobjdef) and                (currobjdef.owner.symtabletype in [globalsymtable,staticsymtable]) and                (currobjdef.owner.iscurrentunit) and                currobjdef.is_related(tobjectdef(owner.defowner))               )           ) then          exit;        is_visible_for_object:=true;      end;    function tprocdef.getsymtable(t:tgetsymtable):tsymtable;      begin        case t of          gs_local :            getsymtable:=localst;          gs_para :            getsymtable:=parast;          else            getsymtable:=nil;        end;      end;    procedure tprocdef.load_references(ppufile:tcompilerppufile;locals:boolean);      var        pos : tfileposinfo;        move_last : boolean;        oldparasymtable,        oldlocalsymtable : tsymtable;      begin        oldparasymtable:=aktparasymtable;        oldlocalsymtable:=aktlocalsymtable;        aktparasymtable:=parast;        aktlocalsymtable:=localst;        move_last:=lastwritten=lastref;        while (not ppufile.endofentry) do         begin           ppufile.getposinfo(pos);           inc(refcount);           lastref:=tref.create(lastref,@pos);           lastref.is_written:=true;           if refcount=1 then            defref:=lastref;         end;        if move_last then          lastwritten:=lastref;        if ((current_module.flags and uf_local_browser)<>0) and           assigned(localst) and           locals then          begin             tparasymtable(parast).load_references(ppufile,locals);             tlocalsymtable(localst).load_references(ppufile,locals);          end;        aktparasymtable:=oldparasymtable;        aktlocalsymtable:=oldlocalsymtable;      end;    Const      local_symtable_index : word = $8001;    function tprocdef.write_references(ppufile:tcompilerppufile;locals:boolean):boolean;      var        ref : tref;{$ifdef supportbrowser}        pdo : tobjectdef;{$endif supportbrowser}        move_last : boolean;        d : tderef;        oldparasymtable,        oldlocalsymtable : tsymtable;      begin        d.reset;        move_last:=lastwritten=lastref;        if move_last and           (((current_module.flags and uf_local_browser)=0) or            not locals) then          exit;        oldparasymtable:=aktparasymtable;        oldlocalsymtable:=aktlocalsymtable;        aktparasymtable:=parast;        aktlocalsymtable:=localst;        { write address of this symbol }        d.build(self);        ppufile.putderef(d);        { write refs }        if assigned(lastwritten) then          ref:=lastwritten        else          ref:=defref;        while assigned(ref) do         begin           if ref.moduleindex=current_module.unit_index then             begin                ppufile.putposinfo(ref.posinfo);                ref.is_written:=true;                if move_last then                  lastwritten:=ref;             end           else if not ref.is_written then             move_last:=false           else if move_last then             lastwritten:=ref;           ref:=ref.nextref;         end;        ppufile.writeentry(ibdefref);        write_references:=true;{$ifdef supportbrowser}        if ((current_module.flags and uf_local_browser)<>0) and           assigned(localst) and           locals then          begin             pdo:=_class;             if (owner.symtabletype<>localsymtable) then               while assigned(pdo) do                 begin                    if pdo.symtable<>aktrecordsymtable then                      begin                         pdo.symtable.moduleid:=local_symtable_index;                         inc(local_symtable_index);                      end;                    pdo:=pdo.childof;                 end;             parast.moduleid:=local_symtable_index;             inc(local_symtable_index);             localst.moduleid:=local_symtable_index;             inc(local_symtable_index);             tstoredsymtable(parast).write_references(ppufile,locals);             tstoredsymtable(localst).write_references(ppufile,locals);             { decrement for }             local_symtable_index:=local_symtable_index-2;             pdo:=_class;             if (owner.symtabletype<>localsymtable) then               while assigned(pdo) do                 begin                    if pdo.symtable<>aktrecordsymtable then                      dec(local_symtable_index);                    pdo:=pdo.childof;                 end;          end;{$endif supportbrowser}        aktparasymtable:=oldparasymtable;        aktlocalsymtable:=oldlocalsymtable;      end;{$ifdef GDB}    function tprocdef.numberstring : string;      begin        { procdefs are always available }        stab_state:=stab_state_written;        result:=inherited numberstring;      end;    function tprocdef.stabstring: pchar;      Var        RType : Char;        Obj,Info : String;        stabsstr : string;        p : pchar;      begin        obj := procsym.name;        info := '';        if tprocsym(procsym).is_global then          RType := 'F'        else          RType := 'f';        if assigned(owner) then         begin           if (owner.symtabletype = objectsymtable) then             obj := owner.name^+'__'+procsym.name;           if not(cs_gdb_valgrind in aktglobalswitches) and              (owner.symtabletype=localsymtable) and              assigned(owner.defowner) and              assigned(tprocdef(owner.defowner).procsym) then             info := ','+procsym.name+','+tprocdef(owner.defowner).procsym.name;         end;        stabsstr:=mangledname;        getmem(p,length(stabsstr)+255);        strpcopy(p,'"'+obj+':'+RType              +tstoreddef(rettype.def).numberstring+info+'",'+tostr(n_function)              +',0,'+              tostr(fileinfo.line)              +',');        strpcopy(strend(p),stabsstr);        stabstring:=strnew(p);        freemem(p,length(stabsstr)+255);      end;    procedure tprocdef.concatstabto(asmlist : taasmoutput);      begin        { released procdef? }        if not assigned(parast) then          exit;        if (proccalloption=pocall_internproc) then          exit;        { be sure to have a number assigned for this def }        numberstring;        { write stabs }        stab_state:=stab_state_writing;        asmList.concat(Tai_stabs.Create(stabstring));        if not(po_external in procoptions) then          begin            tparasymtable(parast).concatstabto(asmlist);            { local type defs and vars should not be written              inside the main proc stab }            if assigned(localst) and               (localst.symtabletype=localsymtable) then              tlocalsymtable(localst).concatstabto(asmlist);          end;        stab_state:=stab_state_written;      end;{$endif GDB}    procedure tprocdef.buildderef;      var        oldparasymtable,        oldlocalsymtable : tsymtable;      begin         oldparasymtable:=aktparasymtable;         oldlocalsymtable:=aktlocalsymtable;         aktparasymtable:=parast;         aktlocalsymtable:=localst;         inherited buildderef;         _classderef.build(_class);         { procsym that originaly defined this definition, should be in the           same symtable }         procsymderef.build(procsym);{$ifdef powerpc}         { library symbol for AmigaOS/MorphOS }         libsymderef.build(libsym);{$endif powerpc}         aktparasymtable:=oldparasymtable;         aktlocalsymtable:=oldlocalsymtable;      end;    procedure tprocdef.buildderefimpl;      var        oldparasymtable,        oldlocalsymtable : tsymtable;      begin         { released procdef? }         if not assigned(parast) then           exit;         oldparasymtable:=aktparasymtable;         oldlocalsymtable:=aktlocalsymtable;         aktparasymtable:=parast;         aktlocalsymtable:=localst;         inherited buildderefimpl;         { Locals }         if assigned(localst) and            ((po_has_inlininginfo in procoptions) or             ((current_module.flags and uf_local_browser)<>0)) then           begin             tlocalsymtable(localst).buildderef;             tlocalsymtable(localst).buildderefimpl;           end;         { inline tree }         if (po_has_inlininginfo in procoptions) then           begin             funcretsymderef.build(funcretsym);             inlininginfo^.code.buildderefimpl;           end;         aktparasymtable:=oldparasymtable;         aktlocalsymtable:=oldlocalsymtable;      end;    procedure tprocdef.deref;      var        oldparasymtable,        oldlocalsymtable : tsymtable;      begin         { released procdef? }         if not assigned(parast) then           exit;         oldparasymtable:=aktparasymtable;         oldlocalsymtable:=aktlocalsymtable;         aktparasymtable:=parast;         aktlocalsymtable:=localst;         inherited deref;         _class:=tobjectdef(_classderef.resolve);         { procsym that originaly defined this definition, should be in the           same symtable }         procsym:=tprocsym(procsymderef.resolve);{$ifdef powerpc}         { library symbol for AmigaOS/MorphOS }         libsym:=tsym(libsymderef.resolve);{$endif powerpc}         aktparasymtable:=oldparasymtable;         aktlocalsymtable:=oldlocalsymtable;      end;    procedure tprocdef.derefimpl;      var        oldparasymtable,        oldlocalsymtable : tsymtable;      begin         oldparasymtable:=aktparasymtable;         oldlocalsymtable:=aktlocalsymtable;         aktparasymtable:=parast;         aktlocalsymtable:=localst;         { 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           include(procoptions,po_has_inlininginfo);         { Locals }         if assigned(localst) then          begin            tlocalsymtable(localst).deref;            tlocalsymtable(localst).derefimpl;          end;        { Inline }        if (po_has_inlininginfo in procoptions) then          begin            inlininginfo^.code.derefimpl;            { funcretsym, this is always located in the localst }            funcretsym:=tsym(funcretsymderef.resolve);          end        else          begin            { safety }            funcretsym:=nil;          end;        aktparasymtable:=oldparasymtable;        aktlocalsymtable:=oldlocalsymtable;      end;    function tprocdef.gettypename : string;      begin         gettypename := FullProcName(false);      end;    function tprocdef.mangledname : string;      var        hp   : TParavarsym;        hs   : string;        crc  : dword;        newlen,        oldlen,        i    : integer;      begin        if assigned(_mangledname) then         begin         {$ifdef compress}           mangledname:=minilzw_decode(_mangledname^);         {$else}           mangledname:=_mangledname^;         {$endif}           exit;         end;        { we need to use the symtable where the procsym is inserted,          because that is visible to the world }        mangledname:=make_mangledname('',procsym.owner,procsym.name);        oldlen:=length(mangledname);        { 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             mangledname:=mangledname+'$'+hp.vartype.def.mangledparaname;         end;        { add resulttype, add $$ as separator to make it unique from a          parameter separator }        if not is_void(rettype.def) then          mangledname:=mangledname+'$$'+rettype.def.mangledparaname;        newlen:=length(mangledname);        { Replace with CRC if the parameter line is very long }        if (newlen-oldlen>12) and           ((newlen>128) or (newlen-oldlen>64)) then          begin            crc:=$ffffffff;            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.vartype.def.mangledparaname;                    crc:=UpdateCrc32(crc,hs[1],length(hs));                  end;              end;            hs:=hp.vartype.def.mangledparaname;            crc:=UpdateCrc32(crc,hs[1],length(hs));            mangledname:=Copy(mangledname,1,oldlen)+'$crc'+hexstr(crc,8);          end;       {$ifdef compress}        _mangledname:=stringdup(minilzw_encode(mangledname));       {$else}        _mangledname:=stringdup(mangledname);       {$endif}      end;    function tprocdef.cplusplusmangledname : string;      function getcppparaname(p : tdef) : string;        const           ordtype2str : array[tbasetype] of string[2] = (             '',             'Uc','Us','Ui','Us',             'Sc','s','i','x',             'b','b','b',             'c','w','x');        var           s : string;        begin           case p.deftype of              orddef:                s:=ordtype2str[torddef(p).typ];              pointerdef:                s:='P'+getcppparaname(tpointerdef(p).pointertype.def);              else                internalerror(2103001);           end;           getcppparaname:=s;        end;      var         s,s2 : string;         hp   : TParavarsym;         i    : integer;      begin         s := procsym.realname;         if procsym.owner.symtabletype=objectsymtable then           begin              s2:=upper(tobjectdef(procsym.owner.defowner).typesym.realname);              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]);                 s2:=getcppparaname(hp.vartype.def);                 if hp.varspez in [vs_var,vs_out] then                   s2:='R'+s2;                 s:=s+s2;               end;           end         else           s:=s+'v';         cplusplusmangledname:=s;      end;    procedure tprocdef.setmangledname(const s : string);      begin        { This is not allowed anymore, the forward declaration          already needs to create the correct mangledname, no changes          afterwards are allowed (PFV) }        if assigned(_mangledname) then          internalerror(200411171);      {$ifdef compress}        _mangledname:=stringdup(minilzw_encode(s));      {$else}        _mangledname:=stringdup(s);      {$endif}        include(procoptions,po_has_mangledname);      end;{***************************************************************************                                 TPROCVARDEF***************************************************************************}    constructor tprocvardef.create(level:byte);      begin         inherited create(level);         deftype:=procvardef;      end;    constructor tprocvardef.ppuload(ppufile:tcompilerppufile);      begin         inherited ppuload(ppufile);         deftype:=procvardef;         { load para symtable }         parast:=tparasymtable.create(unknown_level);         tparasymtable(parast).ppuload(ppufile);         parast.defowner:=self;      end;    function tprocvardef.getcopy : tstoreddef;      begin        result:=self;      (*          { saves a definition to the return type }          rettype         : ttype;          parast          : tsymtable;          paras           : tparalist;          proctypeoption  : tproctypeoption;          proccalloption  : tproccalloption;          procoptions     : tprocoptions;          requiredargarea : aint;          { number of user visibile parameters }          maxparacount,          minparacount    : byte;{$ifdef i386}          fpu_used        : longint;    { how many stack fpu must be empty }{$endif i386}          funcretloc : array[tcallercallee] of TLocation;          has_paraloc_info : boolean; { paraloc info is available }       tprocvardef = class(tabstractprocdef)          constructor create(level:byte);          constructor ppuload(ppufile:tcompilerppufile);          function getcopy : tstoreddef;override;       *)      end;    procedure tprocvardef.ppuwrite(ppufile:tcompilerppufile);      var        oldparasymtable,        oldlocalsymtable : tsymtable;      begin        oldparasymtable:=aktparasymtable;        oldlocalsymtable:=aktlocalsymtable;        aktparasymtable:=parast;        aktlocalsymtable:=nil;        { here we cannot get a real good value so just give something }        { plausible (PM) }        { a more secure way would be          to allways store in a temp }{$ifdef i386}        if is_fpu(rettype.def) then          fpu_used:={2}maxfpuregs        else          fpu_used:=0;{$endif i386}        inherited ppuwrite(ppufile);        { Write this entry }        ppufile.writeentry(ibprocvardef);        { Save the para symtable, this is taken from the interface }        tparasymtable(parast).ppuwrite(ppufile);        aktparasymtable:=oldparasymtable;        aktlocalsymtable:=oldlocalsymtable;      end;    procedure tprocvardef.buildderef;      var        oldparasymtable,        oldlocalsymtable : tsymtable;      begin        oldparasymtable:=aktparasymtable;        oldlocalsymtable:=aktlocalsymtable;        aktparasymtable:=parast;        aktlocalsymtable:=nil;        inherited buildderef;        aktparasymtable:=oldparasymtable;        aktlocalsymtable:=oldlocalsymtable;      end;    procedure tprocvardef.deref;      var        oldparasymtable,        oldlocalsymtable : tsymtable;      begin        oldparasymtable:=aktparasymtable;        oldlocalsymtable:=aktlocalsymtable;        aktparasymtable:=parast;        aktlocalsymtable:=nil;        inherited deref;        aktparasymtable:=oldparasymtable;        aktlocalsymtable:=oldlocalsymtable;      end;    function tprocvardef.getsymtable(t:tgetsymtable):tsymtable;      begin        case t of          gs_para :            getsymtable:=parast;          else            getsymtable:=nil;        end;      end;    function tprocvardef.size : aint;      begin         if (po_methodpointer in procoptions) and            not(po_addressonly in procoptions) then           size:=2*sizeof(aint)         else           size:=sizeof(aint);      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_addressonly in procoptions);      end;    function tprocvardef.getmangledparaname:string;      begin        result:='procvar';      end;{$ifdef GDB}    function tprocvardef.stabstring : pchar;      var         nss : pchar;        { i   : longint; }      begin        { i := maxparacount; }        getmem(nss,1024);        { it is not a function but a function pointer !! (PM) }        strpcopy(nss,'*f'+tstoreddef(rettype.def).numberstring{+','+tostr(i)});        { this confuses gdb !!          we should use 'F' instead of 'f' but          as we use c++ language mode          it does not like that either          Please do not remove this part          might be used once          gdb for pascal is ready PM }      {$ifdef disabled}        param := para1;        i := 0;        while assigned(param) do          begin            inc(i);            if param^.paratyp = vs_value then vartyp := '1' else vartyp := '0';            {Here we have lost the parameter names !!}            pst := strpnew('p'+tostr(i)+':'+param^.vartype.def.numberstring+','+vartyp+';');            strcat(nss,pst);            strdispose(pst);            param := param^.next;          end;      {$endif}        {strpcopy(strend(nss),';');}        stabstring := strnew(nss);        freemem(nss,1024);      end;    procedure tprocvardef.concatstabto(asmlist : taasmoutput);      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;        tstoreddef(rettype.def).concatstabto(asmlist);        inherited concatstabto(asmlist);      end;{$endif GDB}    procedure tprocvardef.write_rtti_data(rt:trttitype);         procedure write_para(parasym:tparavarsym);         var           paraspec : byte;         begin           { only store user visible parameters }           if not(vo_is_hidden_para in parasym.varoptions) then             begin               case parasym.varspez of                 vs_value: paraspec := 0;                 vs_const: paraspec := pfConst;                 vs_var  : paraspec := pfVar;                 vs_out  : paraspec := pfOut;               end;               { write flags for current parameter }               rttiList.concat(Tai_const.Create_8bit(paraspec));               { write name of current parameter }               rttiList.concat(Tai_const.Create_8bit(length(parasym.realname)));               rttiList.concat(Tai_string.Create(parasym.realname));               { write name of type of current parameter }               tstoreddef(parasym.vartype.def).write_rtti_name;             end;         end;       var         methodkind : byte;         i : integer;      begin        if po_methodpointer in procoptions then          begin             { write method id and name }             rttiList.concat(Tai_const.Create_8bit(tkmethod));             write_rtti_name;{$ifdef cpurequiresproperalignment}             rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}             { write kind of method (can only be function or procedure)}             if rettype.def = voidtype.def then               methodkind := mkProcedure             else               methodkind := mkFunction;             rttiList.concat(Tai_const.Create_8bit(methodkind));             { get # of parameters }             rttiList.concat(Tai_const.Create_8bit(maxparacount));             { write parameter info. The parameters must be written in reverse order               if this method uses right to left parameter pushing! }             if proccalloption in pushleftright_pocalls then               begin                 for i:=0 to paras.count-1 do                   write_para(tparavarsym(paras[i]));               end             else               begin                 for i:=paras.count-1 downto 0 do                   write_para(tparavarsym(paras[i]));               end;             { write name of result type }             tstoreddef(rettype.def).write_rtti_name;          end;      end;    function tprocvardef.is_publishable : boolean;      begin         is_publishable:=(po_methodpointer in procoptions);      end;    function tprocvardef.gettypename : string;      var        s: string;        showhidden : boolean;      begin{$ifdef EXTDEBUG}         showhidden:=true;{$else EXTDEBUG}         showhidden:=false;{$endif EXTDEBUG}         s:='<';         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 po_local in procoptions then           s := s+' local';         if assigned(rettype.def) and            (rettype.def<>voidtype.def) then           s:=s+' function'+typename_paras(showhidden)+':'+rettype.def.gettypename         else           s:=s+' procedure'+typename_paras(showhidden);         if po_methodpointer in procoptions then           s := s+' of object';         gettypename := s+';'+ProcCallOptionStr[proccalloption]+'>';      end;{***************************************************************************                              TOBJECTDEF***************************************************************************}   constructor tobjectdef.create(ot : tobjectdeftype;const n : string;c : tobjectdef);     begin        inherited create;        objecttype:=ot;        deftype:=objectdef;        objectoptions:=[];        childof:=nil;        symtable:=tobjectsymtable.create(n,aktpackrecords);        { create space for vmt !! }        vmt_offset:=0;        symtable.defowner:=self;        lastvtableindex:=0;        set_parent(c);        objname:=stringdup(upper(n));        objrealname:=stringdup(n);        if objecttype in [odt_interfacecorba,odt_interfacecom] then          prepareguid;        { setup implemented interfaces }        if objecttype in [odt_class,odt_interfacecorba] then          implementedinterfaces:=timplementedinterfaces.create        else          implementedinterfaces:=nil;{$ifdef GDB}        writing_class_record_stab:=false;{$endif GDB}     end;    constructor tobjectdef.ppuload(ppufile:tcompilerppufile);      var         i,implintfcount: longint;         d : tderef;      begin         inherited ppuloaddef(ppufile);         deftype:=objectdef;         objecttype:=tobjectdeftype(ppufile.getbyte);         objrealname:=stringdup(ppufile.getstring);         objname:=stringdup(upper(objrealname^));         symtable:=tobjectsymtable.create(objrealname^,0);         tobjectsymtable(symtable).datasize:=ppufile.getaint;         tobjectsymtable(symtable).fieldalignment:=ppufile.getbyte;         tobjectsymtable(symtable).recordalignment:=ppufile.getbyte;         vmt_offset:=ppufile.getlongint;         ppufile.getderef(childofderef);         ppufile.getsmallset(objectoptions);         { load guid }         iidstr:=nil;         if objecttype in [odt_interfacecom,odt_interfacecorba] then           begin              new(iidguid);              ppufile.getguid(iidguid^);              iidstr:=stringdup(ppufile.getstring);              lastvtableindex:=ppufile.getlongint;           end;         { load implemented interfaces }         if objecttype in [odt_class,odt_interfacecorba] then           begin             implementedinterfaces:=timplementedinterfaces.create;             implintfcount:=ppufile.getlongint;             for i:=1 to implintfcount do               begin                  ppufile.getderef(d);                  implementedinterfaces.addintf_deref(d,ppufile.getlongint);               end;           end         else           implementedinterfaces:=nil;         tobjectsymtable(symtable).ppuload(ppufile);         symtable.defowner:=self;         { handles the predefined class tobject  }         { the last TOBJECT which is loaded gets }         { it !                                  }         if (childof=nil) and            (objecttype=odt_class) and            (objname^='TOBJECT') then           class_tobject:=self;         if (childof=nil) and            (objecttype=odt_interfacecom) and            (objname^='IUNKNOWN') then           interface_iunknown:=self;{$ifdef GDB}         writing_class_record_stab:=false;{$endif GDB}       end;    destructor tobjectdef.destroy;      begin         if assigned(symtable) then           symtable.free;         stringdispose(objname);         stringdispose(objrealname);         if assigned(iidstr) then           stringdispose(iidstr);         if assigned(implementedinterfaces) then           implementedinterfaces.free;         if assigned(iidguid) then           dispose(iidguid);         inherited destroy;      end;    function tobjectdef.getcopy : tstoreddef;      begin        result:=inherited getcopy;      (*        result:=tobjectdef.create(objecttype,objname^,childof);          childofderef  : tderef;          objname,          objrealname   : pstring;          objectoptions : tobjectoptions;          { to be able to have a variable vmt position }          { and no vmt field for objects without virtuals }          vmt_offset : longint;{$ifdef GDB}          writing_class_record_stab : boolean;{$endif GDB}          objecttype : tobjectdeftype;          iidguid: pguid;          iidstr: pstring;          lastvtableindex: longint;          { store implemented interfaces defs and name mappings }          implementedinterfaces: timplementedinterfaces;      *)      end;    procedure tobjectdef.ppuwrite(ppufile:tcompilerppufile);      var         implintfcount : longint;         i : longint;      begin         inherited ppuwritedef(ppufile);         ppufile.putbyte(byte(objecttype));         ppufile.putstring(objrealname^);         ppufile.putaint(tobjectsymtable(symtable).datasize);         ppufile.putbyte(tobjectsymtable(symtable).fieldalignment);         ppufile.putbyte(tobjectsymtable(symtable).recordalignment);         ppufile.putlongint(vmt_offset);         ppufile.putderef(childofderef);         ppufile.putsmallset(objectoptions);         if objecttype in [odt_interfacecom,odt_interfacecorba] then           begin              ppufile.putguid(iidguid^);              ppufile.putstring(iidstr^);              ppufile.putlongint(lastvtableindex);           end;         if objecttype in [odt_class,odt_interfacecorba] then           begin              implintfcount:=implementedinterfaces.count;              ppufile.putlongint(implintfcount);              for i:=1 to implintfcount do                begin                   ppufile.putderef(implementedinterfaces.interfacesderef(i));                   ppufile.putlongint(implementedinterfaces.ioffsets(i));                end;           end;         ppufile.writeentry(ibobjectdef);         tobjectsymtable(symtable).ppuwrite(ppufile);      end;    function tobjectdef.gettypename:string;      begin        gettypename:=typename;      end;    procedure tobjectdef.buildderef;      var         oldrecsyms : tsymtable;      begin         inherited buildderef;         childofderef.build(childof);         oldrecsyms:=aktrecordsymtable;         aktrecordsymtable:=symtable;         tstoredsymtable(symtable).buildderef;         aktrecordsymtable:=oldrecsyms;         if objecttype in [odt_class,odt_interfacecorba] then           implementedinterfaces.buildderef;      end;    procedure tobjectdef.deref;      var         oldrecsyms : tsymtable;      begin         inherited deref;         childof:=tobjectdef(childofderef.resolve);         oldrecsyms:=aktrecordsymtable;         aktrecordsymtable:=symtable;         tstoredsymtable(symtable).deref;         aktrecordsymtable:=oldrecsyms;         if objecttype in [odt_class,odt_interfacecorba] then           implementedinterfaces.deref;      end;    function tobjectdef.getparentdef:tdef;      begin{$warning 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        { nothing to do if the parent was not forward !}        if assigned(childof) then          exit;        childof:=c;        { some options are inherited !! }        if assigned(c) then          begin             { only important for classes }             lastvtableindex:=c.lastvtableindex;             objectoptions:=objectoptions+(c.objectoptions*               inherited_objectoptions);             if not (objecttype in [odt_interfacecom,odt_interfacecorba]) then               begin                  { add the data of the anchestor class }                  inc(tobjectsymtable(symtable).datasize,tobjectsymtable(c.symtable).datasize);                  if (oo_has_vmt in objectoptions) and                     (oo_has_vmt in c.objectoptions) then                    dec(tobjectsymtable(symtable).datasize,sizeof(aint));                  { 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_offset:=c.vmt_offset;                       include(objectoptions,oo_has_vmt);                    end;               end;          end;      end;   procedure tobjectdef.insertvmt;     begin        if objecttype in [odt_interfacecom,odt_interfacecorba] then          exit;        if (oo_has_vmt in objectoptions) then          internalerror(12345)        else          begin             tobjectsymtable(symtable).datasize:=align(tobjectsymtable(symtable).datasize,                 tobjectsymtable(symtable).fieldalignment);{$ifdef cpurequiresproperalignment}             tobjectsymtable(symtable).datasize:=align(tobjectsymtable(symtable).datasize,sizeof(aint));{$endif cpurequiresproperalignment}             vmt_offset:=tobjectsymtable(symtable).datasize;             inc(tobjectsymtable(symtable).datasize,sizeof(aint));             include(objectoptions,oo_has_vmt);          end;     end;   procedure tobjectdef.check_forwards;     begin        if not(objecttype in [odt_interfacecom,odt_interfacecorba]) then          tstoredsymtable(symtable).check_forwards;        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;   { true, if self inherits from d (or if they are equal) }   function tobjectdef.is_related(d : tdef) : boolean;     var        hp : tobjectdef;     begin        hp:=self;        while assigned(hp) do          begin             if hp=d then               begin                  is_related:=true;                  exit;               end;             hp:=hp.childof;          end;        is_related:=false;     end;(*   procedure tobjectdef._searchdestructor(sym : tnamedindexitem;arg:pointer);     var        p : pprocdeflist;     begin        { if we found already a destructor, then we exit }        if assigned(sd) then          exit;        if tsym(sym).typ=procsym then          begin             p:=tprocsym(sym).defs;             while assigned(p) do               begin                  if p^.def.proctypeoption=potype_destructor then                    begin                       sd:=p^.def;                       exit;                    end;                  p:=p^.next;               end;          end;     end;*)    procedure _searchdestructor(sym:Tnamedindexitem;sd:pointer);    begin        { if we found already a destructor, then we exit }        if (ppointer(sd)^=nil) and           (Tsym(sym).typ=procsym) then          ppointer(sd)^:=Tprocsym(sym).search_procdef_bytype(potype_destructor);    end;   function tobjectdef.searchdestructor : tprocdef;     var        o : tobjectdef;        sd : tprocdef;     begin        searchdestructor:=nil;        o:=self;        sd:=nil;        while assigned(o) do          begin             o.symtable.foreach_static(@_searchdestructor,@sd);             if assigned(sd) then               begin                  searchdestructor:=sd;                  exit;               end;             o:=o.childof;          end;     end;    function tobjectdef.size : aint;      begin        if objecttype in [odt_class,odt_interfacecom,odt_interfacecorba] then          result:=sizeof(aint)        else          result:=tobjectsymtable(symtable).datasize;      end;    function tobjectdef.alignment:longint;      begin        if objecttype in [odt_class,odt_interfacecom,odt_interfacecorba] then          alignment:=sizeof(aint)        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(Aint) is size and -size }          vmtmethodoffset:=(index+10)*sizeof(aint)+2*sizeof(AInt);        odt_interfacecom,odt_interfacecorba:          vmtmethodoffset:=index*sizeof(aint);        else{$ifdef WITHDMT}          vmtmethodoffset:=(index+4)*sizeof(aint);{$else WITHDMT}          vmtmethodoffset:=(index+3)*sizeof(aint);{$endif WITHDMT}        end;      end;    function tobjectdef.vmt_mangledname : string;      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.rtti_name : string;      begin        rtti_name:=make_mangledname('RTTI',owner,objname^);      end;{$ifdef GDB}    procedure tobjectdef.proc_addname(p :tnamedindexitem;arg:pointer);      var virtualind,argnames : string;          newrec : pchar;          pd     : tprocdef;          lindex : longint;          arglength : byte;          sp : char;          state:^Trecord_stabgen_state;          olds:integer;          i : integer;          parasym : tparavarsym;      begin        state:=arg;        if tsym(p).typ = procsym then         begin           pd := tprocsym(p).first_procdef;           if (po_virtualmethod in pd.procoptions) then             begin               lindex := pd.extnumber;               {doesnt seem to be necessary               lindex := lindex or $80000000;}               virtualind := '*'+tostr(lindex)+';'+pd._class.classnumberstring+';'             end            else             virtualind := '.';            { used by gdbpas to recognize constructor and destructors }            if (pd.proctypeoption=potype_constructor) then              argnames:='__ct__'            else if (pd.proctypeoption=potype_destructor) then              argnames:='__dt__'            else              argnames := '';           { arguments are not listed here }           {we don't need another definition}            for i:=0 to pd.paras.count-1 do              begin                parasym:=tparavarsym(pd.paras[i]);                if Parasym.vartype.def.deftype = formaldef then                  begin                    case Parasym.varspez of                      vs_var :                        argnames := argnames+'3var';                      vs_const :                        argnames:=argnames+'5const';                      vs_out :                        argnames:=argnames+'3out';                    end;                  end                else                  begin                    { if the arg definition is like (v: ^byte;..                    there is no sym attached to data !!! }                    if assigned(Parasym.vartype.def.typesym) then                      begin                        arglength := length(Parasym.vartype.def.typesym.name);                        argnames := argnames + tostr(arglength)+Parasym.vartype.def.typesym.name;                      end                    else                      argnames:=argnames+'11unnamedtype';                  end;              end;           { here 2A must be changed for private and protected }           { 0 is private 1 protected and 2 public }           if ([sp_private,sp_strictprivate]*tsym(p).symoptions)<>[] then             sp:='0'           else if ([sp_protected,sp_strictprotected]*tsym(p).symoptions)<>[] then             sp:='1'           else             sp:='2';           newrec:=stabstr_evaluate('$1::$2=##$3;:$4;$5A$6;',[p.name,pd.numberstring,                                    Tstoreddef(pd.rettype.def).numberstring,argnames,sp,                                    virtualind]);           { get spare place for a string at the end }           olds:=state^.stabsize;           inc(state^.stabsize,strlen(newrec));           if state^.stabsize>=state^.staballoc-256 then             begin                inc(state^.staballoc,memsizeinc);                reallocmem(state^.stabstring,state^.staballoc);             end;           strcopy(state^.stabstring+olds,newrec);           strdispose(newrec);           {This should be used for case !!           RecOffset := RecOffset + pd.size;}         end;      end;    procedure tobjectdef.proc_concatstabto(p :tnamedindexitem;arg:pointer);      var        pd : tprocdef;      begin        if tsym(p).typ = procsym then          begin            pd := tprocsym(p).first_procdef;            tstoreddef(pd.rettype.def).concatstabto(taasmoutput(arg));          end;      end;    function tobjectdef.stabstring : pchar;      var anc : tobjectdef;          state:Trecord_stabgen_state;          ts : string;      begin        if not (objecttype=odt_class) or writing_class_record_stab then          begin            state.staballoc:=memsizeinc;            getmem(state.stabstring,state.staballoc);            strpcopy(state.stabstring,'s'+tostr(tobjectsymtable(symtable).datasize));            if assigned(childof) then              begin                {only one ancestor not virtual, public, at base offset 0 }                {       !1           ,    0       2         0    ,       }                strpcopy(strend(state.stabstring),'!1,020,'+childof.classnumberstring+';');              end;            {virtual table to implement yet}            state.recoffset:=0;            state.stabsize:=strlen(state.stabstring);            symtable.foreach(@field_addname,@state);            if (oo_has_vmt in objectoptions) then              if not assigned(childof) or not(oo_has_vmt in childof.objectoptions) then                 begin                    ts:='$vf'+classnumberstring+':'+tstoreddef(vmtarraytype.def).numberstring+','+tostr(vmt_offset*8)+';';                    strpcopy(state.stabstring+state.stabsize,ts);                    inc(state.stabsize,length(ts));                 end;            symtable.foreach(@proc_addname,@state);            if (oo_has_vmt in objectoptions) then              begin                 anc := self;                 while assigned(anc.childof) and (oo_has_vmt in anc.childof.objectoptions) do                   anc := anc.childof;                 { just in case anc = self }                 ts:=';~%'+anc.classnumberstring+';';              end            else              ts:=';';            strpcopy(state.stabstring+state.stabsize,ts);            inc(state.stabsize,length(ts));            reallocmem(state.stabstring,state.stabsize+1);            stabstring:=state.stabstring;          end        else          begin            stabstring:=strpnew('*'+classnumberstring);          end;      end;   procedure tobjectdef.set_globalnb;     begin         globalnb:=PglobalTypeCount^;         inc(PglobalTypeCount^);         { classes need two type numbers, the globalnb is set to the ptr }         if objecttype=odt_class then           begin             globalnb:=PGlobalTypeCount^;             inc(PglobalTypeCount^);           end;     end;   function tobjectdef.classnumberstring : string;     begin       if objecttype=odt_class then         begin           if globalnb=0 then             numberstring;           dec(globalnb);           classnumberstring:=numberstring;           inc(globalnb);         end       else         classnumberstring:=numberstring;     end;    function tobjectdef.allstabstring : pchar;      var        stabchar : string[2];        ss,st : pchar;        sname : string;      begin        ss := stabstring;        getmem(st,strlen(ss)+512);        stabchar := 't';        if deftype in tagtypes then          stabchar := 'Tt';        if assigned(typesym) then          sname := typesym.name        else          sname := ' ';        if writing_class_record_stab then          strpcopy(st,'"'+sname+':'+stabchar+classnumberstring+'=')        else          strpcopy(st,'"'+sname+':'+stabchar+numberstring+'=');        strpcopy(strecopy(strend(st),ss),'",'+tostr(N_LSYM)+',0,0,0');        allstabstring := strnew(st);        freemem(st,strlen(ss)+512);        strdispose(ss);      end;    procedure tobjectdef.concatstabto(asmlist : taasmoutput);      var        oldtypesym : tsym;        stab_str   : pchar;        anc        : tobjectdef;      begin        if (stab_state in [stab_state_writing,stab_state_written]) then          exit;        stab_state:=stab_state_writing;        tstoreddef(vmtarraytype.def).concatstabto(asmlist);        { first the parents }        anc:=self;        while assigned(anc.childof) do          begin            anc:=anc.childof;            anc.concatstabto(asmlist);          end;        symtable.foreach(@field_concatstabto,asmlist);        symtable.foreach(@proc_concatstabto,asmlist);        stab_state:=stab_state_used;        if objecttype=odt_class then          begin            { Write the record class itself }            writing_class_record_stab:=true;            inherited concatstabto(asmlist);            writing_class_record_stab:=false;            { Write the invisible pointer class }            oldtypesym:=typesym;            typesym:=nil;            stab_str := allstabstring;            asmList.concat(Tai_stabs.Create(stab_str));            typesym:=oldtypesym;          end        else          inherited concatstabto(asmlist);      end;{$endif GDB}    function tobjectdef.needs_inittable : boolean;      begin         case objecttype of            odt_class :              needs_inittable:=false;            odt_interfacecom:              needs_inittable:=true;            odt_interfacecorba:              needs_inittable:=is_related(interface_iunknown);            odt_object:              needs_inittable:=tobjectsymtable(symtable).needs_init_final;            else              internalerror(200108267);         end;      end;    function tobjectdef.members_need_inittable : boolean;      begin        members_need_inittable:=tobjectsymtable(symtable).needs_init_final;      end;    procedure tobjectdef.count_published_properties(sym:tnamedindexitem;arg:pointer);      begin         if needs_prop_entry(tsym(sym)) and            (tsym(sym).typ<>fieldvarsym) then           inc(count);      end;    procedure tobjectdef.write_property_info(sym : tnamedindexitem;arg:pointer);      var         proctypesinfo : byte;      procedure writeproc(proc : tsymlist; shiftvalue : byte);        var           typvalue : byte;           hp : psymlistitem;           address : longint;           def : tdef;        begin           if not(assigned(proc) and assigned(proc.firstsym))  then             begin                rttiList.concat(Tai_const.create(ait_const_ptr,1));                typvalue:=3;             end           else if proc.firstsym^.sym.typ=fieldvarsym then             begin                address:=0;                hp:=proc.firstsym;                def:=nil;                while assigned(hp) do                  begin                     case hp^.sltype of                       sl_load :                         begin                           def:=tfieldvarsym(hp^.sym).vartype.def;                           inc(address,tfieldvarsym(hp^.sym).fieldoffset);                         end;                       sl_subscript :                         begin                           if not(assigned(def) and (def.deftype=recorddef)) then                             internalerror(200402171);                           inc(address,tfieldvarsym(hp^.sym).fieldoffset);                           def:=tfieldvarsym(hp^.sym).vartype.def;                         end;                       sl_vec :                         begin                           if not(assigned(def) and (def.deftype=arraydef)) then                             internalerror(200402172);                           def:=tarraydef(def).elementtype.def;                           inc(address,def.size*hp^.value);                         end;                     end;                     hp:=hp^.next;                  end;                rttiList.concat(Tai_const.create(ait_const_ptr,address));                typvalue:=0;             end           else             begin                { When there was an error then procdef is not assigned }                if not assigned(proc.procdef) then                  exit;                if not(po_virtualmethod in tprocdef(proc.procdef).procoptions) then                  begin                     rttiList.concat(Tai_const.createname(tprocdef(proc.procdef).mangledname,AT_FUNCTION,0));                     typvalue:=1;                  end                else                  begin                     { virtual method, write vmt offset }                     rttiList.concat(Tai_const.create(ait_const_ptr,                       tprocdef(proc.procdef)._class.vmtmethodoffset(tprocdef(proc.procdef).extnumber)));                     typvalue:=2;                  end;             end;           proctypesinfo:=proctypesinfo or (typvalue shl shiftvalue);        end;      begin         if needs_prop_entry(tsym(sym)) then           case tsym(sym).typ of              fieldvarsym:                begin{$ifdef dummy}                   if not(tvarsym(sym).vartype.def.deftype=objectdef) or                     not(tobjectdef(tvarsym(sym).vartype.def).is_class) then                     internalerror(1509992);                   { access to implicit class property as field }                   proctypesinfo:=(0 shl 0) or (0 shl 2) or (0 shl 4);                   rttiList.concat(Tai_const_symbol.Createname(tvarsym(sym.vartype.def.get_rtti_label),AT_DATA,0));                   rttiList.concat(Tai_const.create(ait_const_ptr,tvarsym(sym.address)));                   rttiList.concat(Tai_const.create(ait_const_ptr,tvarsym(sym.address)));                   { by default stored }                   rttiList.concat(Tai_const.Create_32bit(1));                   { index as well as ... }                   rttiList.concat(Tai_const.Create_32bit(0));                   { default value are zero }                   rttiList.concat(Tai_const.Create_32bit(0));                   rttiList.concat(Tai_const.Create_16bit(count));                   inc(count);                   rttiList.concat(Tai_const.Create_8bit(proctypesinfo));                   rttiList.concat(Tai_const.Create_8bit(length(tvarsym(sym.realname))));                   rttiList.concat(Tai_string.Create(tvarsym(sym.realname)));{$endif dummy}                end;              propertysym:                begin                   if ppo_indexed in tpropertysym(sym).propoptions then                     proctypesinfo:=$40                   else                     proctypesinfo:=0;                   rttiList.concat(Tai_const.Create_sym(tstoreddef(tpropertysym(sym).proptype.def).get_rtti_label(fullrtti)));                   writeproc(tpropertysym(sym).readaccess,0);                   writeproc(tpropertysym(sym).writeaccess,2);                   { isn't it stored ? }                   if not(ppo_stored in tpropertysym(sym).propoptions) then                     begin                        rttiList.concat(Tai_const.create_sym(nil));                        proctypesinfo:=proctypesinfo or (3 shl 4);                     end                   else                     writeproc(tpropertysym(sym).storedaccess,4);                   rttiList.concat(Tai_const.Create_32bit(tpropertysym(sym).index));                   rttiList.concat(Tai_const.Create_32bit(tpropertysym(sym).default));                   rttiList.concat(Tai_const.Create_16bit(count));                   inc(count);                   rttiList.concat(Tai_const.Create_8bit(proctypesinfo));                   rttiList.concat(Tai_const.Create_8bit(length(tpropertysym(sym).realname)));                   rttiList.concat(Tai_string.Create(tpropertysym(sym).realname));{$ifdef cpurequiresproperalignment}                   rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}                end;              else internalerror(1509992);           end;      end;    procedure tobjectdef.generate_published_child_rtti(sym : tnamedindexitem;arg:pointer);      begin         if needs_prop_entry(tsym(sym)) then          begin            case tsym(sym).typ of              propertysym:                tstoreddef(tpropertysym(sym).proptype.def).get_rtti_label(fullrtti);              fieldvarsym:                tstoreddef(tfieldvarsym(sym).vartype.def).get_rtti_label(fullrtti);              else                internalerror(1509991);            end;          end;      end;    procedure tobjectdef.write_child_rtti_data(rt:trttitype);      begin         FRTTIType:=rt;         case rt of           initrtti :             symtable.foreach(@generate_field_rtti,nil);           fullrtti :             symtable.foreach(@generate_published_child_rtti,nil);           else             internalerror(200108301);         end;      end;    type       tclasslistitem = class(TLinkedListItem)          index : longint;          p : tobjectdef;       end;    var       classtablelist : tlinkedlist;       tablecount : longint;    function searchclasstablelist(p : tobjectdef) : tclasslistitem;      var         hp : tclasslistitem;      begin         hp:=tclasslistitem(classtablelist.first);         while assigned(hp) do           if hp.p=p then             begin                searchclasstablelist:=hp;                exit;             end           else             hp:=tclasslistitem(hp.next);         searchclasstablelist:=nil;      end;    procedure tobjectdef.count_published_fields(sym:tnamedindexitem;arg:pointer);      var         hp : tclasslistitem;      begin         if needs_prop_entry(tsym(sym)) and          (tsym(sym).typ=fieldvarsym) then          begin             if tfieldvarsym(sym).vartype.def.deftype<>objectdef then               internalerror(0206001);             hp:=searchclasstablelist(tobjectdef(tfieldvarsym(sym).vartype.def));             if not(assigned(hp)) then               begin                  hp:=tclasslistitem.create;                  hp.p:=tobjectdef(tfieldvarsym(sym).vartype.def);                  hp.index:=tablecount;                  classtablelist.concat(hp);                  inc(tablecount);               end;             inc(count);          end;      end;    procedure tobjectdef.writefields(sym:tnamedindexitem;arg:pointer);      var         hp : tclasslistitem;      begin         if needs_prop_entry(tsym(sym)) and          (tsym(sym).typ=fieldvarsym) then          begin{$ifdef cpurequiresproperalignment}             rttilist.concat(Tai_align.Create(sizeof(AInt)));{$endif cpurequiresproperalignment}             rttiList.concat(Tai_const.Create_aint(tfieldvarsym(sym).fieldoffset));             hp:=searchclasstablelist(tobjectdef(tfieldvarsym(sym).vartype.def));             if not(assigned(hp)) then               internalerror(0206002);             rttiList.concat(Tai_const.Create_16bit(hp.index));             rttiList.concat(Tai_const.Create_8bit(length(tfieldvarsym(sym).realname)));             rttiList.concat(Tai_string.Create(tfieldvarsym(sym).realname));          end;      end;    function tobjectdef.generate_field_table : tasmlabel;      var         fieldtable,         classtable : tasmlabel;         hp : tclasslistitem;      begin         classtablelist:=TLinkedList.Create;         objectlibrary.getdatalabel(fieldtable);         objectlibrary.getdatalabel(classtable);         count:=0;         tablecount:=0;         maybe_new_object_file(rttiList);         new_section(rttiList,sec_rodata,classtable.name,const_align(sizeof(aint)));         { fields }         symtable.foreach({$ifdef FPC}@{$endif}count_published_fields,nil);         rttiList.concat(Tai_label.Create(fieldtable));         rttiList.concat(Tai_const.Create_16bit(count));{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}         rttiList.concat(Tai_const.Create_sym(classtable));         symtable.foreach({$ifdef FPC}@{$endif}writefields,nil);         { generate the class table }         rttilist.concat(tai_align.create(const_align(sizeof(aint))));         rttiList.concat(Tai_label.Create(classtable));         rttiList.concat(Tai_const.Create_16bit(tablecount));{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}         hp:=tclasslistitem(classtablelist.first);         while assigned(hp) do           begin              rttiList.concat(Tai_const.Createname(tobjectdef(hp.p).vmt_mangledname,AT_DATA,0));              hp:=tclasslistitem(hp.next);           end;         generate_field_table:=fieldtable;         classtablelist.free;      end;    function tobjectdef.next_free_name_index : longint;      var         i : longint;      begin         if assigned(childof) and (oo_can_have_published in childof.objectoptions) then           i:=childof.next_free_name_index         else           i:=0;         count:=0;         symtable.foreach(@count_published_properties,nil);         next_free_name_index:=i+count;      end;    procedure tobjectdef.write_rtti_data(rt:trttitype);      var        i : longint;      begin         case objecttype of            odt_class:              rttiList.concat(Tai_const.Create_8bit(tkclass));            odt_object:              rttiList.concat(Tai_const.Create_8bit(tkobject));            odt_interfacecom:              rttiList.concat(Tai_const.Create_8bit(tkinterface));            odt_interfacecorba:              rttiList.concat(Tai_const.Create_8bit(tkinterfaceCorba));          else            exit;          end;         { generate the name }         rttiList.concat(Tai_const.Create_8bit(length(objrealname^)));         rttiList.concat(Tai_string.Create(objrealname^));{$ifdef cpurequiresproperalignment}         rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}         case rt of           initrtti :             begin               rttiList.concat(Tai_const.Create_32bit(size));               if objecttype in [odt_class,odt_object] then                begin                  count:=0;                  FRTTIType:=rt;                  symtable.foreach(@count_field_rtti,nil);                  rttiList.concat(Tai_const.Create_32bit(count));                  symtable.foreach(@write_field_rtti,nil);                end;             end;           fullrtti :             begin               if not(objecttype in [odt_interfacecom,odt_interfacecorba]) then                 begin                   if (oo_has_vmt in objectoptions) then                     rttiList.concat(Tai_const.Createname(vmt_mangledname,AT_DATA,0))                   else                     rttiList.concat(Tai_const.create_sym(nil));                 end;               { write parent typeinfo }               if assigned(childof) and ((oo_can_have_published in childof.objectoptions) or                 (objecttype in [odt_interfacecom,odt_interfacecorba])) then                 rttiList.concat(Tai_const.Create_sym(childof.get_rtti_label(fullrtti)))               else                 rttiList.concat(Tai_const.create_sym(nil));               if objecttype in [odt_object,odt_class] then                 begin                   { count total number of properties }                   if assigned(childof) and (oo_can_have_published in childof.objectoptions) then                     count:=childof.next_free_name_index                   else                     count:=0;                   { write it }                   symtable.foreach(@count_published_properties,nil);                   rttiList.concat(Tai_const.Create_16bit(count));                 end               else                 { interface: write flags, iid and iidstr }                 begin                   rttiList.concat(Tai_const.Create_32bit(                     { ugly, but working }                     longint([                       TCompilerIntfFlag(ord(ifHasGuid)*ord(assigned(iidguid))),                       TCompilerIntfFlag(ord(ifHasStrGUID)*ord(assigned(iidstr)))                     ])                     {                     ifDispInterface,                     ifDispatch, }                     ));{$ifdef cpurequiresproperalignment}                   rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}                   rttilist.concat(Tai_const.Create_32bit(longint(iidguid^.D1)));                   rttilist.concat(Tai_const.Create_16bit(iidguid^.D2));                   rttilist.concat(Tai_const.Create_16bit(iidguid^.D3));                   for i:=Low(iidguid^.D4) to High(iidguid^.D4) do                     rttilist.concat(Tai_const.Create_8bit(iidguid^.D4[i]));                 end;               { write unit name }               rttiList.concat(Tai_const.Create_8bit(length(current_module.realmodulename^)));               rttiList.concat(Tai_string.Create(current_module.realmodulename^));{$ifdef cpurequiresproperalignment}               rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}               { write iidstr }               if objecttype in [odt_interfacecom,odt_interfacecorba] then                 begin                   if assigned(iidstr) then                     begin                       rttiList.concat(Tai_const.Create_8bit(length(iidstr^)));                       rttiList.concat(Tai_string.Create(iidstr^));                     end                   else                     rttiList.concat(Tai_const.Create_8bit(0));{$ifdef cpurequiresproperalignment}                   rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}                 end;               if objecttype in [odt_object,odt_class] then                 begin                   { write published properties count }                   count:=0;                   symtable.foreach(@count_published_properties,nil);                   rttiList.concat(Tai_const.Create_16bit(count));{$ifdef cpurequiresproperalignment}                   rttilist.concat(Tai_align.Create(sizeof(TConstPtrUInt)));{$endif cpurequiresproperalignment}                 end;               { count is used to write nameindex   }               { but we need an offset of the owner }               { to give each property an own slot  }               if assigned(childof) and (oo_can_have_published in childof.objectoptions) then                 count:=childof.next_free_name_index               else                 count:=0;               symtable.foreach(@write_property_info,nil);             end;         end;      end;    function tobjectdef.is_publishable : boolean;      begin         is_publishable:=objecttype in [odt_class,odt_interfacecom,odt_interfacecorba];      end;{****************************************************************************                             TIMPLEMENTEDINTERFACES****************************************************************************}    type      tnamemap = class(TNamedIndexItem)        newname: pstring;        constructor create(const aname, anewname: string);        destructor  destroy; override;      end;    constructor tnamemap.create(const aname, anewname: string);      begin        inherited createname(name);        newname:=stringdup(anewname);      end;    destructor  tnamemap.destroy;      begin        stringdispose(newname);        inherited destroy;      end;    type      tprocdefstore = class(TNamedIndexItem)        procdef: tprocdef;        constructor create(aprocdef: tprocdef);      end;    constructor tprocdefstore.create(aprocdef: tprocdef);      begin        inherited create;        procdef:=aprocdef;      end;    constructor timplintfentry.create(aintf: tobjectdef);      begin        inherited create;        intf:=aintf;        ioffset:=-1;        namemappings:=nil;        procdefs:=nil;      end;    constructor timplintfentry.create_deref(const d:tderef);      begin        inherited create;        intf:=nil;        intfderef:=d;        ioffset:=-1;        namemappings:=nil;        procdefs:=nil;      end;    destructor  timplintfentry.destroy;      begin        if assigned(namemappings) then          namemappings.free;        if assigned(procdefs) then          procdefs.free;        inherited destroy;      end;    constructor timplementedinterfaces.create;      begin        finterfaces:=tindexarray.create(1);      end;    destructor  timplementedinterfaces.destroy;      begin        finterfaces.destroy;      end;    function  timplementedinterfaces.count: longint;      begin        count:=finterfaces.count;      end;    procedure timplementedinterfaces.checkindex(intfindex: longint);      begin        if (intfindex<1) or (intfindex>count) then          InternalError(200006123);      end;    function  timplementedinterfaces.interfaces(intfindex: longint): tobjectdef;      begin        checkindex(intfindex);        interfaces:=timplintfentry(finterfaces.search(intfindex)).intf;      end;    function  timplementedinterfaces.interfacesderef(intfindex: longint): tderef;      begin        checkindex(intfindex);        interfacesderef:=timplintfentry(finterfaces.search(intfindex)).intfderef;      end;    function  timplementedinterfaces.ioffsets(intfindex: longint): longint;      begin        checkindex(intfindex);        ioffsets:=timplintfentry(finterfaces.search(intfindex)).ioffset;      end;    procedure timplementedinterfaces.setioffsets(intfindex,iofs:longint);      begin        checkindex(intfindex);        timplintfentry(finterfaces.search(intfindex)).ioffset:=iofs;      end;    function timplementedinterfaces.implindex(intfindex:longint):longint;      begin        checkindex(intfindex);        result:=timplintfentry(finterfaces.search(intfindex)).implindex;      end;    procedure timplementedinterfaces.setimplindex(intfindex,implidx:longint);      begin        checkindex(intfindex);        timplintfentry(finterfaces.search(intfindex)).implindex:=implidx;      end;    function  timplementedinterfaces.searchintf(def: tdef): longint;      var        i: longint;      begin        i:=1;        while (i<=count) and (tdef(interfaces(i))<>def) do inc(i);        if i<=count then          searchintf:=i        else          searchintf:=-1;      end;    procedure timplementedinterfaces.buildderef;      var        i: longint;      begin        for i:=1 to count do          with timplintfentry(finterfaces.search(i)) do            intfderef.build(intf);      end;    procedure timplementedinterfaces.deref;      var        i: longint;      begin        for i:=1 to count do          with timplintfentry(finterfaces.search(i)) do            intf:=tobjectdef(intfderef.resolve);      end;    procedure timplementedinterfaces.addintf_deref(const d:tderef;iofs:longint);      var        hintf : timplintfentry;      begin        hintf:=timplintfentry.create_deref(d);        hintf.ioffset:=iofs;        finterfaces.insert(hintf);      end;    procedure timplementedinterfaces.addintf(def: tdef);      begin        if not assigned(def) or (searchintf(def)<>-1) or (def.deftype<>objectdef) or           not (tobjectdef(def).objecttype in [odt_interfacecom,odt_interfacecorba]) then          internalerror(200006124);        finterfaces.insert(timplintfentry.create(tobjectdef(def)));      end;    procedure timplementedinterfaces.clearmappings;      var        i: longint;      begin        for i:=1 to count do          with timplintfentry(finterfaces.search(i)) do            begin              if assigned(namemappings) then                namemappings.free;              namemappings:=nil;            end;      end;    procedure timplementedinterfaces.addmappings(intfindex: longint; const name, newname: string);      begin        checkindex(intfindex);        with timplintfentry(finterfaces.search(intfindex)) do          begin            if not assigned(namemappings) then              namemappings:=tdictionary.create;            namemappings.insert(tnamemap.create(name,newname));          end;      end;    function  timplementedinterfaces.getmappings(intfindex: longint; const name: string; var nextexist: pointer): string;      begin        checkindex(intfindex);        if not assigned(nextexist) then          with timplintfentry(finterfaces.search(intfindex)) do            begin              if assigned(namemappings) then                nextexist:=namemappings.search(name)              else                nextexist:=nil;            end;        if assigned(nextexist) then          begin            getmappings:=tnamemap(nextexist).newname^;            nextexist:=tnamemap(nextexist).listnext;          end        else          getmappings:='';      end;    procedure timplementedinterfaces.addimplproc(intfindex: longint; procdef: tprocdef);      var        found : boolean;        i     : longint;      begin        checkindex(intfindex);        with timplintfentry(finterfaces.search(intfindex)) do          begin            if not assigned(procdefs) then              procdefs:=tindexarray.create(4);            { No duplicate entries of the same procdef }            found:=false;            for i:=1 to procdefs.count do              if tprocdefstore(procdefs.search(i)).procdef=procdef then                begin                  found:=true;                  break;                end;            if not found then              procdefs.insert(tprocdefstore.create(procdef));          end;      end;    function  timplementedinterfaces.implproccount(intfindex: longint): longint;      begin        checkindex(intfindex);        with timplintfentry(finterfaces.search(intfindex)) do          if assigned(procdefs) then            implproccount:=procdefs.count          else            implproccount:=0;      end;    function  timplementedinterfaces.implprocs(intfindex: longint; procindex: longint): tprocdef;      begin        checkindex(intfindex);        with timplintfentry(finterfaces.search(intfindex)) do          if assigned(procdefs) then            implprocs:=tprocdefstore(procdefs.search(procindex)).procdef          else            internalerror(200006131);      end;    function  timplementedinterfaces.isimplmergepossible(intfindex, remainindex: longint; var weight: longint): boolean;      var        possible: boolean;        i: longint;        iiep1: TIndexArray;        iiep2: TIndexArray;      begin        checkindex(intfindex);        checkindex(remainindex);        iiep1:=timplintfentry(finterfaces.search(intfindex)).procdefs;        iiep2:=timplintfentry(finterfaces.search(remainindex)).procdefs;        if not assigned(iiep1) then { empty interface is mergeable :-) }          begin            possible:=true;            weight:=0;          end        else          begin            possible:=assigned(iiep2) and (iiep1.count<=iiep2.count);            i:=1;            while (possible) and (i<=iiep1.count) do              begin                possible:=                  (tprocdefstore(iiep1.search(i)).procdef=tprocdefstore(iiep2.search(i)).procdef);                inc(i);              end;            if possible then              weight:=iiep1.count;          end;        isimplmergepossible:=possible;      end;{****************************************************************************                                TFORWARDDEF****************************************************************************}   constructor tforwarddef.create(const s:string;const pos : tfileposinfo);     var       oldregisterdef : boolean;     begin        { never register the forwarddefs, they are disposed at the          end of the type declaration block }        oldregisterdef:=registerdef;        registerdef:=false;        inherited create;        registerdef:=oldregisterdef;        deftype:=forwarddef;        tosymname:=stringdup(s);        forwardpos:=pos;     end;    function tforwarddef.gettypename:string;      begin        gettypename:='unresolved forward to '+tosymname^;      end;     destructor tforwarddef.destroy;      begin        if assigned(tosymname) then          stringdispose(tosymname);        inherited destroy;      end;{****************************************************************************                                  TERRORDEF****************************************************************************}   constructor terrordef.create;     begin        inherited create;        deftype:=errordef;     end;    procedure terrordef.ppuwrite(ppufile:tcompilerppufile);      begin        { Can't write errordefs to ppu }        internalerror(200411063);      end;{$ifdef GDB}    function terrordef.stabstring : pchar;      begin         stabstring:=strpnew('error'+numberstring);      end;    procedure terrordef.concatstabto(asmlist : taasmoutput);      begin        { No internal error needed, an normal error is already          thrown }      end;{$endif GDB}    function terrordef.gettypename:string;      begin         gettypename:='<erroneous type>';      end;    function terrordef.getmangledparaname:string;      begin         getmangledparaname:='error';      end;{****************************************************************************                           Definition Helpers****************************************************************************}    function is_interfacecom(def: tdef): boolean;      begin        is_interfacecom:=          assigned(def) and          (def.deftype=objectdef) and          (tobjectdef(def).objecttype=odt_interfacecom);      end;    function is_interfacecorba(def: tdef): boolean;      begin        is_interfacecorba:=          assigned(def) and          (def.deftype=objectdef) and          (tobjectdef(def).objecttype=odt_interfacecorba);      end;    function is_interface(def: tdef): boolean;      begin        is_interface:=          assigned(def) and          (def.deftype=objectdef) and          (tobjectdef(def).objecttype in [odt_interfacecom,odt_interfacecorba]);      end;    function is_class(def: tdef): boolean;      begin        is_class:=          assigned(def) and          (def.deftype=objectdef) and          (tobjectdef(def).objecttype=odt_class);      end;    function is_object(def: tdef): boolean;      begin        is_object:=          assigned(def) and          (def.deftype=objectdef) and          (tobjectdef(def).objecttype=odt_object);      end;    function is_cppclass(def: tdef): boolean;      begin        is_cppclass:=          assigned(def) and          (def.deftype=objectdef) and          (tobjectdef(def).objecttype=odt_cppclass);      end;    function is_class_or_interface(def: tdef): boolean;      begin        is_class_or_interface:=          assigned(def) and          (def.deftype=objectdef) and          (tobjectdef(def).objecttype in [odt_class,odt_interfacecom,odt_interfacecorba]);      end;end.
 |