| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881 | {    Copyright (c) 2015 by Nikolay Nikolov    Contains the binary Relocatable Object Module Format (OMF) reader and writer    This is the object format used on the i8086-msdos platform.    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 ogomf;{$i fpcdefs.inc}{$PackSet 1}interface    uses       { common }       cclasses,globtype,       { target }       systems,       { assembler }       cpuinfo,cpubase,aasmbase,assemble,link,       { OMF definitions }       omfbase,       { output }       ogbase,       owbase;    type      { TOmfObjSymbol }      TOmfObjSymbol = class(TObjSymbol)      public        { string representation for the linker map file }        function AddressStr(AImageBase: qword): string;override;      end;      { TOmfRelocation }      TOmfRelocation = class(TObjRelocation)      private        FFrameGroup: string;        FOmfFixup: TOmfSubRecord_FIXUP;      public        destructor Destroy; override;        procedure BuildOmfFixup;        property FrameGroup: string read FFrameGroup write FFrameGroup;        property OmfFixup: TOmfSubRecord_FIXUP read FOmfFixup;      end;      TMZExeUnifiedLogicalSegment=class;      { TOmfObjSection }      TOmfObjSection = class(TObjSection)      private        FClassName: string;        FOverlayName: string;        FFirstSym: TObjSymbol;        FCombination: TOmfSegmentCombination;        FUse: TOmfSegmentUse;        FPrimaryGroup: TObjSectionGroup;        FSortOrder: Integer;        FMZExeUnifiedLogicalSegment: TMZExeUnifiedLogicalSegment;        FLinNumEntries: TOmfSubRecord_LINNUM_MsLink_LineNumberList;        function GetOmfAlignment: TOmfSegmentAlignment;      protected        function GetAltName: string; override;      public        constructor create(AList:TFPHashObjectList;const Aname:string;Aalign:longint;Aoptions:TObjSectionOptions);override;        destructor destroy;override;        function MemPosStr(AImageBase: qword): string;override;        property ClassName: string read FClassName;        property OverlayName: string read FOverlayName;        property OmfAlignment: TOmfSegmentAlignment read GetOmfAlignment;        property Combination: TOmfSegmentCombination read FCombination;        property Use: TOmfSegmentUse read FUse;        property PrimaryGroup: TObjSectionGroup read FPrimaryGroup;        property SortOrder: Integer read FSortOrder write FSortOrder;        property MZExeUnifiedLogicalSegment: TMZExeUnifiedLogicalSegment read FMZExeUnifiedLogicalSegment write FMZExeUnifiedLogicalSegment;        property LinNumEntries: TOmfSubRecord_LINNUM_MsLink_LineNumberList read FLinNumEntries;      end;      { TOmfObjExportedSymbol }      TOmfObjExportedSymbol = class(TFPHashObject)      private        FExportByOrdinal: Boolean;        FResidentName: Boolean;        FNoData: Boolean;        FParmCount: Integer;        FExportedName: string;        FInternalName: string;        FExportOrdinal: Word;      public        property ExportByOrdinal: Boolean read FExportByOrdinal write FExportByOrdinal;        property ResidentName: Boolean read FResidentName write FResidentName;        property NoData: Boolean read FNoData write FNoData;        property ParmCount: Integer read FParmCount write FParmCount;        property ExportedName: string read FExportedName write FExportedName;        property InternalName: string read FInternalName write FInternalName;        property ExportOrdinal: Word read FExportOrdinal write FExportOrdinal;      end;      { TOmfObjData }      TOmfObjData = class(TObjData)      private        FMainSource: TPathStr;        FImportLibraryList:TFPHashObjectList;        FExportedSymbolList:TFPHashObjectList;        class function CodeSectionName(const aname:string): string;      public        constructor create(const n:string);override;        destructor destroy;override;        function sectiontype2align(atype:TAsmSectiontype):longint;override;        class function sectiontype2class(atype:TAsmSectiontype):string;        function sectionname(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder):string;override;        function createsection(atype:TAsmSectionType;const aname:string='';aorder:TAsmSectionOrder=secorder_default):TObjSection;override;        function reffardatasection:TObjSection;        procedure writeReloc(Data:TRelocDataInt;len:aword;p:TObjSymbol;Reloctype:TObjRelocationType);override;        procedure AddImportSymbol(const libname,symname,symmangledname:TCmdStr;OrdNr: longint;isvar:boolean);        procedure AddExportSymbol(aExportByOrdinal,aResidentName,aNoData:Boolean;aParmCount:Integer;aExportedName,aInternalName:string;aExportOrdinal:Word);        property MainSource: TPathStr read FMainSource;        property ImportLibraryList:TFPHashObjectList read FImportLibraryList;        property ExportedSymbolList:TFPHashObjectList read FExportedSymbolList;      end;      { TOmfObjOutput }      TOmfObjOutput = class(tObjOutput)      private        FLNames: TOmfOrderedNameCollection;        FSegments: TFPHashObjectList;        FGroups: TFPHashObjectList;        procedure AddSegment(const name,segclass,ovlname: string;          Alignment: TOmfSegmentAlignment; Combination: TOmfSegmentCombination;          Use: TOmfSegmentUse; Size: TObjSectionOfs);        procedure AddGroup(group: TObjSectionGroup);        procedure WriteSections(Data:TObjData);        procedure WriteSectionContentAndFixups(sec: TObjSection);        procedure WriteLinNumRecords(sec: TOmfObjSection);        procedure section_count_sections(p:TObject;arg:pointer);        procedure group_count_groups(p:TObject;arg:pointer);        procedure WritePUBDEFs(Data: TObjData);        procedure WriteEXTDEFs(Data: TObjData);        property LNames: TOmfOrderedNameCollection read FLNames;        property Segments: TFPHashObjectList read FSegments;        property Groups: TFPHashObjectList read FGroups;      protected        function writeData(Data:TObjData):boolean;override;      public        constructor create(AWriter:TObjectWriter);override;        destructor Destroy;override;        procedure WriteDllImport(const dllname,afuncname,mangledname:string;ordnr:longint;isvar:boolean);      end;      { TOmfObjInput }      TOmfObjInput = class(TObjInput)      private        FLNames: TOmfOrderedNameCollection;        FExtDefs: TFPHashObjectList;        FPubDefs: TFPHashObjectList;        FFixupThreads: TOmfThreads;        FRawRecord: TOmfRawRecord;        FCOMENTRecord: TOmfRecord_COMENT;        FCaseSensitiveSegments: Boolean;        FCaseSensitiveSymbols: Boolean;        function PeekNextRecordType: Byte;        function ReadLNames(RawRec: TOmfRawRecord): Boolean;        function ReadSegDef(RawRec: TOmfRawRecord; objdata:TObjData): Boolean;        function ReadGrpDef(RawRec: TOmfRawRecord; objdata:TObjData): Boolean;        function ReadExtDef(RawRec: TOmfRawRecord; objdata:TObjData): Boolean;        function ReadPubDef(RawRec: TOmfRawRecord; objdata:TObjData): Boolean;        function ReadModEnd(RawRec: TOmfRawRecord; objdata:TObjData): Boolean;        function ReadLeOrLiDataAndFixups(RawRec: TOmfRawRecord; objdata:TObjData): Boolean;        function ReadImpDef(Rec: TOmfRecord_COMENT; objdata:TObjData): Boolean;        function ReadExpDef(Rec: TOmfRecord_COMENT; objdata:TObjData): Boolean;        function ImportOmfFixup(objdata: TObjData; objsec: TOmfObjSection; Fixup: TOmfSubRecord_FIXUP): Boolean;        property LNames: TOmfOrderedNameCollection read FLNames;        property ExtDefs: TFPHashObjectList read FExtDefs;        property PubDefs: TFPHashObjectList read FPubDefs;        { Specifies whether we're case sensitive in regards to segment, class, overlay and group names. }        property CaseSensitiveSegments: Boolean read FCaseSensitiveSegments write FCaseSensitiveSegments;        { Specifies whether symbol names (in EXTDEF and PUBDEF records) are case sensitive. }        property CaseSensitiveSymbols: Boolean read FCaseSensitiveSymbols write FCaseSensitiveSymbols;      public        constructor create;override;        destructor destroy;override;        class function CanReadObjData(AReader:TObjectreader):boolean;override;        function ReadObjData(AReader:TObjectreader;out objdata:TObjData):boolean;override;      end;      { TMZExeRelocation }      TMZExeRelocation = record        offset: Word;        segment: Word;      end;      TMZExeRelocations = array of TMZExeRelocation;      TMZExeExtraHeaderData = array of Byte;      { TMZExeHeader }      TMZExeHeader = class      private        FChecksum: Word;        FExtraHeaderData: TMZExeExtraHeaderData;        FHeaderSizeAlignment: Integer;        FInitialCS: Word;        FInitialIP: Word;        FInitialSP: Word;        FInitialSS: Word;        FLoadableImageSize: DWord;        FMaxExtraParagraphs: Word;        FMinExtraParagraphs: Word;        FOverlayNumber: Word;        FRelocations: TMZExeRelocations;        procedure SetHeaderSizeAlignment(AValue: Integer);      public        constructor Create;        procedure WriteTo(aWriter: TObjectWriter);        procedure AddRelocation(aSegment,aOffset: Word);        property HeaderSizeAlignment: Integer read FHeaderSizeAlignment write SetHeaderSizeAlignment; {default=16, must be multiple of 16}        property Relocations: TMZExeRelocations read FRelocations write FRelocations;        property ExtraHeaderData: TMZExeExtraHeaderData read FExtraHeaderData write FExtraHeaderData;        property LoadableImageSize: DWord read FLoadableImageSize write FLoadableImageSize;        property MinExtraParagraphs: Word read FMinExtraParagraphs write FMinExtraParagraphs;        property MaxExtraParagraphs: Word read FMaxExtraParagraphs write FMaxExtraParagraphs;        property InitialSS: Word read FInitialSS write FInitialSS;        property InitialSP: Word read FInitialSP write FInitialSP;        property Checksum: Word read FChecksum write FChecksum;        property InitialIP: Word read FInitialIP write FInitialIP;        property InitialCS: Word read FInitialCS write FInitialCS;        property OverlayNumber: Word read FOverlayNumber write FOverlayNumber;      end;      { TMZExeSection }      TMZExeSection=class(TExeSection)      public        procedure AddObjSection(objsec:TObjSection;ignoreprops:boolean=false);override;      end;      { TMZExeUnifiedLogicalSegment }      TMZExeUnifiedLogicalSegment=class(TFPHashObject)      private        FObjSectionList: TFPObjectList;        FSegName: TSymStr;        FSegClass: TSymStr;        FPrimaryGroup: string;      public        Size,        MemPos,        MemBasePos: qword;        IsStack: Boolean;        constructor create(HashObjectList:TFPHashObjectList;const s:TSymStr);        destructor destroy;override;        procedure AddObjSection(ObjSec: TOmfObjSection);        procedure CalcMemPos;        function MemPosStr:string;        property ObjSectionList: TFPObjectList read FObjSectionList;        property SegName: TSymStr read FSegName;        property SegClass: TSymStr read FSegClass;        property PrimaryGroup: string read FPrimaryGroup write FPrimaryGroup;      end;      { TMZExeUnifiedLogicalGroup }      TMZExeUnifiedLogicalGroup=class(TFPHashObject)      private        FSegmentList: TFPHashObjectList;      public        Size,        MemPos: qword;        constructor create(HashObjectList:TFPHashObjectList;const s:TSymStr);        destructor destroy;override;        procedure CalcMemPos;        function MemPosStr:string;        procedure AddSegment(UniSeg: TMZExeUnifiedLogicalSegment);        property SegmentList: TFPHashObjectList read FSegmentList;      end;      { TMZExeOutput }      TMZExeOutput = class(TExeOutput)      private        FMZFlatContentSection: TMZExeSection;        FExeUnifiedLogicalSegments: TFPHashObjectList;        FExeUnifiedLogicalGroups: TFPHashObjectList;        FDwarfUnifiedLogicalSegments: TFPHashObjectList;        FHeader: TMZExeHeader;        function GetMZFlatContentSection: TMZExeSection;        procedure CalcDwarfUnifiedLogicalSegmentsForSection(const SecName: TSymStr);        procedure CalcExeUnifiedLogicalSegments;        procedure CalcExeGroups;        procedure CalcSegments_MemBasePos;        procedure WriteMap_SegmentsAndGroups;        procedure WriteMap_HeaderData;        function FindStackSegment: TMZExeUnifiedLogicalSegment;        procedure FillLoadableImageSize;        procedure FillMinExtraParagraphs;        procedure FillMaxExtraParagraphs;        procedure FillStartAddress;        procedure FillStackAddress;        procedure FillHeaderData;        function writeExe:boolean;        function writeCom:boolean;        function writeDebugElf:boolean;        property ExeUnifiedLogicalSegments: TFPHashObjectList read FExeUnifiedLogicalSegments;        property ExeUnifiedLogicalGroups: TFPHashObjectList read FExeUnifiedLogicalGroups;        property DwarfUnifiedLogicalSegments: TFPHashObjectList read FExeUnifiedLogicalSegments;        property Header: TMZExeHeader read FHeader;      protected        procedure DoRelocationFixup(objsec:TObjSection);override;        procedure Order_ObjSectionList(ObjSectionList : TFPObjectList;const aPattern:string);override;        function writeData:boolean;override;      public        constructor create;override;        destructor destroy;override;        procedure Load_Symbol(const aname:string);override;        procedure MemPos_EndExeSection;override;        procedure MemPos_ExeSection(const aname:string);override;        property MZFlatContentSection: TMZExeSection read GetMZFlatContentSection;      end;    const      NewExeHeaderSize = $40;      NewExeSegmentHeaderSize = 8;      NewExeRelocationRecordSize = 8;    type      TNewExeHeaderFlag = (        nehfSingleData,                                               { bit  0 }        nehfMultipleData,                                             { bit  1 }        { 'Global initialization' according to BP7's TDUMP.EXE                 }        nehfRealMode,                                                 { bit  2 }        nehfProtectedModeOnly,                                        { bit  3 }        { 'EMSDIRECT' according to OpenWatcom's wdump                          }        { '8086 instructions' according to Ralf Brown's Interrupt List         }        nehfReserved4,                                                { bit  4 }        { 'EMSBANK' according to OpenWatcom's wdump                            }        { '80286 instructions' according to Ralf Brown's Interrupt List        }        nehfReserved5,                                                { bit  5 }        { 'EMSGLOBAL' according to OpenWatcom's wdump                          }        { '80386 instructions' according to Ralf Brown's Interrupt List        }        nehfReserved6,                                                { bit  6 }        nehfNeedsFPU,                                                 { bit  7 }        { Not compatible with windowing API                                    }        nehfNotWindowAPICompatible,                                   { bit  8 }        { Compatible with windowing API                                        }        { (NotWindowAPICompatible + WindowAPICompatible) = Uses windowing API  }        nehfWindowAPICompatible,                                      { bit  9 }        { Family Application (OS/2) according to Ralf Brown's Interrupt List   }        nehfReserved10,                                               { bit 10 }        nehfSelfLoading,                                              { bit 11 }        nehfReserved12,                                               { bit 12 }        nehfLinkErrors,                                               { bit 13 }        nehfReserved14,                                               { bit 14 }        nehfIsDLL);                                                   { bit 15 }      TNewExeHeaderFlags = set of TNewExeHeaderFlag;      TNewExeAdditionalHeaderFlag = (        neahfLFNSupport,                                              { bit  0 }        neahfWindows2ProtectedMode,                                   { bit  1 }        neahfWindows2ProportionalFonts,                               { bit  2 }        neahfHasGangloadArea);                                        { bit  3 }      TNewExeAdditionalHeaderFlags = set of TNewExeAdditionalHeaderFlag;      TNewExeTargetOS = (        netoUnknown                        = $00,        netoOS2                            = $01,        netoWindows                        = $02,        netoMultitaskingMsDos4             = $03,        netoWindows386                     = $04,        netoBorlandOperatingSystemServices = $05,        netoPharLap286DosExtenderOS2       = $81,        netoPharLap286DosExtenderWindows   = $82);      TNewExeSegmentFlag = (        nesfData,                                                     { bit  0 }        nesfLoaderAllocatedMemory,                                    { bit  1 }        nesfLoaded,                                                   { bit  2 }        nesfReserved3,                                                { bit  3 }        nesfMovable,                                                  { bit  4 }        nesfShareable,                                                { bit  5 }        nesfPreload,                                                  { bit  6 }        nesfExecuteOnlyCodeOrReadOnlyData,                            { bit  7 }        nesfHasRelocationData,                                        { bit  8 }        nesfReserved9,                                                { bit  9 }        nesfReserved10,                                               { bit 10 }        nesfReserved11,                                               { bit 11 }        nesfDiscardable,                                              { bit 12 }        nesfReserved13,                                               { bit 13 }        nesfReserved14,                                               { bit 14 }        nesfReserved15);                                              { bit 15 }      TNewExeSegmentFlags = set of TNewExeSegmentFlag;      TNewExeMsDosStub = array of byte;      { TNewExeHeader }      TNewExeHeader = class      private        FMsDosStub: TNewExeMsDosStub;        FLinkerVersion: Byte;        FLinkerRevision: Byte;        FEntryTableOffset: Word;        FEntryTableLength: Word;        FReserved: LongWord;        FFlags: TNewExeHeaderFlags;        FAutoDataSegmentNumber: Word;        FInitialLocalHeapSize: Word;        FInitialStackSize: Word;        FInitialIP: Word;        FInitialCS: Word;        FInitialSP: Word;        FInitialSS: Word;        FSegmentTableEntriesCount: Word;        FModuleReferenceTableEntriesCount: Word;        FNonresidentNameTableLength: Word;        FSegmentTableStart: Word;        FResourceTableStart: Word;        FResidentNameTableStart: Word;        FModuleReferenceTableStart: Word;        FImportedNameTableStart: Word;        FNonresidentNameTableStart: LongWord;        FMovableEntryPointsCount: Word;        FLogicalSectorAlignmentShiftCount: Word;        FResourceSegmentsCount: Word;        FTargetOS: TNewExeTargetOS;        FAdditionalFlags: TNewExeAdditionalHeaderFlags;        FGangLoadAreaStart: Word;        FGangLoadAreaLength: Word;        FReserved2: Word;        FExpectedWindowsVersion: Word;      public        constructor Create;        procedure WriteTo(aWriter: TObjectWriter);        property MsDosStub: TNewExeMsDosStub read FMsDosStub write FMsDosStub;        property LinkerVersion: Byte read FLinkerVersion write FLinkerVersion;        property LinkerRevision: Byte read FLinkerRevision write FLinkerRevision;        property EntryTableOffset: Word read FEntryTableOffset write FEntryTableOffset;        property EntryTableLength: Word read FEntryTableLength write FEntryTableLength;        property Reserved: LongWord read FReserved write FReserved;        property Flags: TNewExeHeaderFlags read FFlags write FFlags;        property AutoDataSegmentNumber: Word read FAutoDataSegmentNumber write FAutoDataSegmentNumber;        property InitialLocalHeapSize: Word read FInitialLocalHeapSize write FInitialLocalHeapSize;        property InitialStackSize: Word read FInitialStackSize write FInitialStackSize;        property InitialIP: Word read FInitialIP write FInitialIP;        property InitialCS: Word read FInitialCS write FInitialCS;        property InitialSP: Word read FInitialSP write FInitialSP;        property InitialSS: Word read FInitialSS write FInitialSS;        property SegmentTableEntriesCount: Word read FSegmentTableEntriesCount write FSegmentTableEntriesCount;        property ModuleReferenceTableEntriesCount: Word read FModuleReferenceTableEntriesCount write FModuleReferenceTableEntriesCount;        property NonresidentNameTableLength: Word read FNonresidentNameTableLength write FNonresidentNameTableLength;        property SegmentTableStart: Word read FSegmentTableStart write FSegmentTableStart;        property ResourceTableStart: Word read FResourceTableStart write FResourceTableStart;        property ResidentNameTableStart: Word read FResidentNameTableStart write FResidentNameTableStart;        property ModuleReferenceTableStart: Word read FModuleReferenceTableStart write FModuleReferenceTableStart;        property ImportedNameTableStart: Word read FImportedNameTableStart write FImportedNameTableStart;        property NonresidentNameTableStart: LongWord read FNonresidentNameTableStart write FNonresidentNameTableStart;        property MovableEntryPointsCount: Word read FMovableEntryPointsCount write FMovableEntryPointsCount;        property LogicalSectorAlignmentShiftCount: Word read FLogicalSectorAlignmentShiftCount write FLogicalSectorAlignmentShiftCount;        property ResourceSegmentsCount: Word read FResourceSegmentsCount write FResourceSegmentsCount;        property TargetOS: TNewExeTargetOS read FTargetOS write FTargetOS;        property AdditionalFlags: TNewExeAdditionalHeaderFlags read FAdditionalFlags write FAdditionalFlags;        property GangLoadAreaStart: Word read FGangLoadAreaStart write FGangLoadAreaStart;        property GangLoadAreaLength: Word read FGangLoadAreaLength write FGangLoadAreaLength;        property Reserved2: Word read FReserved2 write FReserved2;        property ExpectedWindowsVersion: Word read FExpectedWindowsVersion write FExpectedWindowsVersion;      end;      { TNewExeResourceTable }      TNewExeResourceTable = class      private        FResourceDataAlignmentShiftCount: Word;        function GetSize: QWord;      public        constructor Create;        procedure WriteTo(aWriter: TObjectWriter);        property ResourceDataAlignmentShiftCount: Word read FResourceDataAlignmentShiftCount write FResourceDataAlignmentShiftCount;        property Size: QWord read GetSize;      end;      { TNewExeExportNameTableEntry }      TNewExeExportNameTableEntry = class(TFPHashObject)      private        FOrdinalNr: Word;      public        constructor Create(HashObjectList:TFPHashObjectList;const s:TSymStr;OrdNr:Word);        property OrdinalNr: Word read FOrdinalNr write FOrdinalNr;      end;      { TNewExeExportNameTable }      TNewExeExportNameTable = class(TFPHashObjectList)      private        function GetSize: QWord;      public        procedure WriteTo(aWriter: TObjectWriter);        property Size: QWord read GetSize;      end;      TNewExeImportedNameTable = class;      { TNewExeModuleReferenceTableEntry }      TNewExeModuleReferenceTableEntry = class(TFPHashObject)      end;      { TNewExeModuleReferenceTable }      TNewExeModuleReferenceTable = class(TFPHashObjectList)      private        function GetSize: QWord;      public        procedure AddModuleReference(const dllname:TSymStr);        procedure WriteTo(aWriter: TObjectWriter;imptbl:TNewExeImportedNameTable);        property Size: QWord read GetSize;      end;      { TNewExeImportedNameTableEntry }      TNewExeImportedNameTableEntry = class(TFPHashObject)      private        FTableOffset: Word;      public        property TableOffset: Word read FTableOffset write FTableOffset;      end;      { TNewExeImportedNameTable }      TNewExeImportedNameTable = class(TFPHashObjectList)      private        function GetSize: QWord;      public        procedure AddImportedName(const name:TSymStr);        procedure CalcTableOffsets;        procedure WriteTo(aWriter: TObjectWriter);        property Size: QWord read GetSize;      end;      TNewExeEntryPointFlag = (        neepfMovableSegment,        neepfExported,        neepfSingleData      );      TNewExeEntryPointFlags = set of TNewExeEntryPointFlag;      { TNewExeEntryPoint }      TNewExeEntryPoint = class      private        FFlags: TNewExeEntryPointFlags;        FSegment: Byte;        FOffset: Word;        FParmCount: Integer;        function GetFlagsByte: Byte;      public        property Flags: TNewExeEntryPointFlags read FFlags write FFlags;        property Segment: Byte read FSegment write FSegment;        property Offset: Word read FOffset write FOffset;        property ParmCount: Integer read FParmCount write FParmCount;        property FlagsByte: Byte read GetFlagsByte;      end;      { TNewExeEntryTable }      TNewExeEntryTable = class      strict private        FItems: array of TNewExeEntryPoint;        function GetCount: Word;        function GetItems(i: Integer): TNewExeEntryPoint;        function GetSize: QWord;        procedure SetItems(i: Integer; AValue: TNewExeEntryPoint);        function CanBeInSameBundle(i,j:Integer):Boolean;        function BundleSize(StartingElement:Integer): Byte;      public        destructor Destroy;override;        procedure WriteTo(aWriter: TObjectWriter);        procedure GrowTo(aNewCount: Word);        property Size: QWord read GetSize;        property Count: Word read GetCount;        property Items[i: Integer]: TNewExeEntryPoint read GetItems write SetItems;default;      end;      { These are fake "meta sections" used by the linker script. The actual        NewExe sections are segments, limited to 64kb, which means there can be        multiple code segments, etc. These are created manually as object        sections are added. If they fit the current segment, without exceeding        64kb, they are added to the current segment, otherwise a new segment is        created. The current "meta sections" tells what kind of new segment to        create (e.g. nemsCode means that a new code segment will be created). }      TNewExeMetaSection = (        nemsNone,        nemsCode,        nemsData);    const      NewExeMetaSection2String: array [TNewExeMetaSection] of string[9] = (        '',        'Code',        'Data');    type      TNewExeRelocationAddressType = (        neratLoByte       = 0,    { low 8 bits of 16-bit offset }        neratSelector     = 2,    { 16-bit selector }        neratFarPointer   = 3,    { 16-bit selector:16-bit offset }        neratOffset       = 5,    { 16-bit offset }        neratFarPointer48 = 11,   { 16-bit selector:32-bit offset }        neratOffset32     = 13);  { 32-bit offset }      TNewExeRelocationType = (        nertInternalRef,        nertImportName,        nertImportOrdinal,        nertOsFixup);      TNewExeOsFixupType = (        neoftFIARQQ_FJARQQ = 1,        neoftFISRQQ_FJSRQQ = 2,        neoftFICRQQ_FJCRQQ = 3,        neoftFIERQQ        = 4,        neoftFIDRQQ        = 5,        neoftFIWRQQ        = 6);      TNewExeInternalRefSegmentType = (        neirstFixed,        neirstMovable);      { TNewExeRelocation }      TNewExeRelocation=class      private        FAddressType: TNewExeRelocationAddressType;        FRelocationType: TNewExeRelocationType;        FIsAdditive: Boolean;        FInternalRefSegmentType: TNewExeInternalRefSegmentType;        FOsFixupType: TNewExeOsFixupType;        FOffset: Word;        FImportModuleIndex: Word;        FImportNameIndex: Word;        FImportOrdinal: Word;        FInternalRefFixedSegmentNumber: Byte;        FInternalRefFixedSegmentOffset: Word;        FInternalRefMovableSegmentEntryTableIndex: Word;      public        procedure EncodeTo(dest: PByte);        property AddressType: TNewExeRelocationAddressType read FAddressType write FAddressType;        property RelocationType: TNewExeRelocationType read FRelocationType write FRelocationType;        property IsAdditive: Boolean read FIsAdditive write FIsAdditive;        property InternalRefSegmentType: TNewExeInternalRefSegmentType read FInternalRefSegmentType write FInternalRefSegmentType;        property OsFixupType: TNewExeOsFixupType read FOsFixupType write FOsFixupType;        property Offset: Word read FOffset write FOffset;        property ImportModuleIndex: Word read FImportModuleIndex write FImportModuleIndex;        property ImportNameIndex: Word read FImportNameIndex write FImportNameIndex;        property ImportOrdinal: Word read FImportOrdinal write FImportOrdinal;        property InternalRefFixedSegmentNumber: Byte read FInternalRefFixedSegmentNumber write FInternalRefFixedSegmentNumber;        property InternalRefFixedSegmentOffset: Word read FInternalRefFixedSegmentOffset write FInternalRefFixedSegmentOffset;        property InternalRefMovableSegmentEntryTableIndex: Word read FInternalRefMovableSegmentEntryTableIndex write FInternalRefMovableSegmentEntryTableIndex;      end;      { TNewExeRelocationList }      TNewExeRelocationList=class      private        FInternalList: TFPObjectList;        function GetCount: Integer;        function GetItem(Index: Integer): TNewExeRelocation;        function GetSize: QWord;        procedure SetCount(AValue: Integer);        procedure SetItem(Index: Integer; AValue: TNewExeRelocation);      public        constructor Create;        destructor Destroy; override;        procedure WriteTo(aWriter: TObjectWriter);        function Add(AObject: TNewExeRelocation): Integer;        property Size: QWord read GetSize;        property Count: Integer read GetCount write SetCount;        property Items[Index: Integer]: TNewExeRelocation read GetItem write SetItem; default;      end;      { TNewExeSection }      TNewExeSection=class(TExeSection)      private        FEarlySize: QWord;        FStackSize: QWord;        FExeMetaSec: TNewExeMetaSection;        FMemBasePos: Word;        FDataPosSectors: Word;        FNewExeSegmentFlags: TNewExeSegmentFlags;        FSizeInFile: QWord;        FRelocations: TNewExeRelocationList;        function GetMinAllocSize: QWord;        function GetNewExeSegmentFlags: TNewExeSegmentFlags;      public        constructor create(AList:TFPHashObjectList;const AName:string);override;        destructor destroy;override;        procedure WriteHeaderTo(aWriter: TObjectWriter);        function MemPosStr(AImageBase: qword): string;override;        procedure AddObjSection(objsec:TObjSection;ignoreprops:boolean=false);override;        function CanAddObjSection(objsec:TObjSection;ExeSectionLimit:QWord):boolean;        property EarlySize: QWord read FEarlySize write FEarlySize;        property StackSize: QWord read FStackSize write FStackSize;        property ExeMetaSec: TNewExeMetaSection read FExeMetaSec write FExeMetaSec;        property MemBasePos: Word read FMemBasePos write FMemBasePos;        property DataPosSectors: Word read FDataPosSectors write FDataPosSectors;        property MinAllocSize: QWord read GetMinAllocSize;        property SizeInFile: QWord read FSizeInFile write FSizeInFile;        property NewExeSegmentFlags: TNewExeSegmentFlags read GetNewExeSegmentFlags write FNewExeSegmentFlags;        property Relocations: TNewExeRelocationList read FRelocations;      end;      { TNewExeOutput }      TNewExeOutput = class(TExeOutput)      private        FHeader: TNewExeHeader;        FImports: TFPHashObjectList;        FCurrExeMetaSec: TNewExeMetaSection;        FResourceTable: TNewExeResourceTable;        FResidentNameTable: TNewExeExportNameTable;        FNonresidentNameTable: TNewExeExportNameTable;        FModuleReferenceTable: TNewExeModuleReferenceTable;        FImportedNameTable: TNewExeImportedNameTable;        FEntryTable: TNewExeEntryTable;        procedure AddImportSymbol(const libname,symname,symmangledname:TCmdStr;OrdNr: longint;isvar:boolean);        procedure AddImportLibrariesExtractedFromObjectModules;        procedure AddNewExeSection;        function WriteNewExe:boolean;        procedure FillImportedNameAndModuleReferenceTable;        function GetHighestExportSymbolOrdinal: Word;        procedure AssignOrdinalsToAllExportSymbols;        procedure AddEntryPointsForAllExportSymbols;        procedure AddExportedNames;        property Header: TNewExeHeader read FHeader;        property CurrExeMetaSec: TNewExeMetaSection read FCurrExeMetaSec write FCurrExeMetaSec;        property ResourceTable: TNewExeResourceTable read FResourceTable;        property ResidentNameTable: TNewExeExportNameTable read FResidentNameTable;        property NonresidentNameTable: TNewExeExportNameTable read FNonresidentNameTable;        property ModuleReferenceTable: TNewExeModuleReferenceTable read FModuleReferenceTable;        property ImportedNameTable: TNewExeImportedNameTable read FImportedNameTable;        property EntryTable: TNewExeEntryTable read FEntryTable;      protected        procedure DoRelocationFixup(objsec:TObjSection);override;        procedure Order_ObjSectionList(ObjSectionList : TFPObjectList;const aPattern:string);override;      public        constructor create;override;        destructor destroy;override;        procedure Order_ExeSection(const aname:string);override;        procedure Order_EndExeSection;override;        procedure Order_ObjSection(const aname:string);override;        procedure MemPos_Start;override;        procedure GenerateLibraryImports(ImportLibraryList:TFPHashObjectList);override;        function writeData:boolean;override;      end;      TOmfAssembler = class(tinternalassembler)        constructor create(info: pasminfo; smart:boolean);override;      end;    function StripDllExt(const DllName:TSymStr):TSymStr;    function MaybeAddDllExt(const DllName:TSymStr):TSymStr;implementation    uses       SysUtils,       cutils,verbose,globals,fpchash,       fmodule,aasmtai,aasmdata,       ogmap,owomflib,elfbase,       version       ;    const win16stub : array[0..255] of byte=(      $4d,$5a,$00,$01,$01,$00,$00,$00,$08,$00,$10,$00,$ff,$ff,$08,$00,      $00,$01,$00,$00,$00,$00,$00,$00,$40,$00,$00,$00,$00,$00,$00,$00,      $00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,      $00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$01,$00,$00,      $00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,      $00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,      $00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,      $00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,$00,      $ba,$10,$00,$0e,$1f,$b4,$09,$cd,$21,$b8,$01,$4c,$cd,$21,$90,$90,      $54,$68,$69,$73,$20,$70,$72,$6f,$67,$72,$61,$6d,$20,$72,$65,$71,      $75,$69,$72,$65,$73,$20,$4d,$69,$63,$72,$6f,$73,$6f,$66,$74,$20,      $57,$69,$6e,$64,$6f,$77,$73,$2e,$0d,$0a,$24,$20,$20,$20,$20,$20,      $20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,      $20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,      $20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,      $20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20,$20);{****************************************************************************                                TTISTrailer****************************************************************************}    const      TIS_TRAILER_SIGNATURE: array[1..4] of char='TIS'#0;      TIS_TRAILER_VENDOR_TIS=0;      TIS_TRAILER_TYPE_TIS_DWARF=0;    type      TTISTrailer=record        tis_signature: array[1..4] of char;        tis_vendor,        tis_type,        tis_size: LongWord;      end;    procedure MayBeSwapTISTrailer(var h: TTISTrailer);      begin        if source_info.endian<>target_info.endian then          with h do            begin              tis_vendor:=swapendian(tis_vendor);              tis_type:=swapendian(tis_type);              tis_size:=swapendian(tis_size);            end;      end;{****************************************************************************                                TOmfObjSymbol****************************************************************************}    function TOmfObjSymbol.AddressStr(AImageBase: qword): string;      var        base: qword;      begin        if assigned(objsection.ExeSection) and (objsection.ExeSection is TNewExeSection) then          Result:=HexStr(TNewExeSection(objsection.ExeSection).MemBasePos,4)+':'+HexStr(address,4)        else          begin            if assigned(TOmfObjSection(objsection).MZExeUnifiedLogicalSegment) then              base:=TOmfObjSection(objsection).MZExeUnifiedLogicalSegment.MemBasePos            else              base:=(address shr 4) shl 4;            Result:=HexStr(base shr 4,4)+':'+HexStr(address-base,4);          end;      end;{****************************************************************************                                TOmfRelocation****************************************************************************}    destructor TOmfRelocation.Destroy;      begin        FOmfFixup.Free;        inherited Destroy;      end;    procedure TOmfRelocation.BuildOmfFixup;      begin        FreeAndNil(FOmfFixup);        FOmfFixup:=TOmfSubRecord_FIXUP.Create;        if ObjSection<>nil then          begin            FOmfFixup.LocationOffset:=DataOffset;            if typ in [RELOC_ABSOLUTE16,RELOC_RELATIVE16] then              FOmfFixup.LocationType:=fltOffset            else if typ in [RELOC_ABSOLUTE32,RELOC_RELATIVE32] then              FOmfFixup.LocationType:=fltOffset32            else if typ in [RELOC_SEG,RELOC_SEGREL] then              FOmfFixup.LocationType:=fltBase            else              internalerror(2015041501);            FOmfFixup.FrameDeterminedByThread:=False;            FOmfFixup.TargetDeterminedByThread:=False;            if typ in [RELOC_ABSOLUTE16,RELOC_ABSOLUTE32,RELOC_SEG] then              FOmfFixup.Mode:=fmSegmentRelative            else if typ in [RELOC_RELATIVE16,RELOC_RELATIVE32,RELOC_SEGREL] then              FOmfFixup.Mode:=fmSelfRelative            else              internalerror(2015041408);            if typ in [RELOC_ABSOLUTE16,RELOC_ABSOLUTE32,RELOC_RELATIVE16,RELOC_RELATIVE32] then              begin                FOmfFixup.TargetMethod:=ftmSegmentIndexNoDisp;                FOmfFixup.TargetDatum:=ObjSection.Index;                if TOmfObjSection(ObjSection).PrimaryGroup<>nil then                  begin                    FOmfFixup.FrameMethod:=ffmGroupIndex;                    FOmfFixup.FrameDatum:=TOmfObjSection(ObjSection).PrimaryGroup.index;                  end                else                  FOmfFixup.FrameMethod:=ffmTarget;              end            else              begin                FOmfFixup.FrameMethod:=ffmTarget;                if TOmfObjSection(ObjSection).PrimaryGroup<>nil then                  begin                    FOmfFixup.TargetMethod:=ftmGroupIndexNoDisp;                    FOmfFixup.TargetDatum:=TOmfObjSection(ObjSection).PrimaryGroup.index;                  end                else                  begin                    FOmfFixup.TargetMethod:=ftmSegmentIndexNoDisp;                    FOmfFixup.TargetDatum:=ObjSection.Index;                  end;              end;          end        else if symbol<>nil then          begin            FOmfFixup.LocationOffset:=DataOffset;            if typ in [RELOC_ABSOLUTE16,RELOC_RELATIVE16] then              FOmfFixup.LocationType:=fltOffset            else if typ in [RELOC_ABSOLUTE32,RELOC_RELATIVE32] then              FOmfFixup.LocationType:=fltOffset32            else if typ in [RELOC_SEG,RELOC_SEGREL] then              FOmfFixup.LocationType:=fltBase            else              internalerror(2015041505);            FOmfFixup.FrameDeterminedByThread:=False;            FOmfFixup.TargetDeterminedByThread:=False;            if typ in [RELOC_ABSOLUTE16,RELOC_ABSOLUTE32,RELOC_SEG] then              FOmfFixup.Mode:=fmSegmentRelative            else if typ in [RELOC_RELATIVE16,RELOC_RELATIVE32,RELOC_SEGREL] then              FOmfFixup.Mode:=fmSelfRelative            else              internalerror(2015041409);            FOmfFixup.TargetMethod:=ftmExternalIndexNoDisp;            FOmfFixup.TargetDatum:=symbol.symidx;            FOmfFixup.FrameMethod:=ffmTarget;          end        else if group<>nil then          begin            FOmfFixup.LocationOffset:=DataOffset;            if typ in [RELOC_ABSOLUTE16,RELOC_RELATIVE16] then              FOmfFixup.LocationType:=fltOffset            else if typ in [RELOC_ABSOLUTE32,RELOC_RELATIVE32] then              FOmfFixup.LocationType:=fltOffset32            else if typ in [RELOC_SEG,RELOC_SEGREL] then              FOmfFixup.LocationType:=fltBase            else              internalerror(2015041506);            FOmfFixup.FrameDeterminedByThread:=False;            FOmfFixup.TargetDeterminedByThread:=False;            if typ in [RELOC_ABSOLUTE16,RELOC_ABSOLUTE32,RELOC_SEG] then              FOmfFixup.Mode:=fmSegmentRelative            else if typ in [RELOC_RELATIVE16,RELOC_RELATIVE32,RELOC_SEGREL] then              FOmfFixup.Mode:=fmSelfRelative            else              internalerror(2015041410);            FOmfFixup.FrameMethod:=ffmTarget;            FOmfFixup.TargetMethod:=ftmGroupIndexNoDisp;            FOmfFixup.TargetDatum:=group.index;          end        else         internalerror(2015040702);      end;{****************************************************************************                                TOmfObjSection****************************************************************************}    function TOmfObjSection.GetOmfAlignment: TOmfSegmentAlignment;      begin        case SecAlign of          1:            result:=saRelocatableByteAligned;          2:            result:=saRelocatableWordAligned;          4:            result:=saRelocatableDWordAligned;          16:            result:=saRelocatableParaAligned;          256:            result:=saRelocatablePageAligned;          4096:            result:=saNotSupported;          else            internalerror(2015041504);        end;      end;    function TOmfObjSection.GetAltName: string;      begin        if FFirstSym<>nil then          result:='/'+FFirstSym.Name        else          result:='';      end;    constructor TOmfObjSection.create(AList: TFPHashObjectList;          const Aname: string; Aalign: longint; Aoptions: TObjSectionOptions);      begin        inherited create(AList, Aname, Aalign, Aoptions);        FCombination:=scPublic;        FUse:=suUse16;        FLinNumEntries:=TOmfSubRecord_LINNUM_MsLink_LineNumberList.Create;      end;    destructor TOmfObjSection.destroy;      begin        FLinNumEntries.Free;        inherited destroy;      end;    function TOmfObjSection.MemPosStr(AImageBase: qword): string;      begin        if Assigned(MZExeUnifiedLogicalSegment) then          Result:=HexStr(MZExeUnifiedLogicalSegment.MemBasePos shr 4,4)+':'+            HexStr(MemPos-MZExeUnifiedLogicalSegment.MemBasePos,4)        else if Assigned(ExeSection) and (ExeSection is TNewExeSection) then          Result:=HexStr(TNewExeSection(ExeSection).MemBasePos,4)+':'+HexStr(mempos,4)        else          Result:=inherited;      end;{****************************************************************************                                TOmfObjData****************************************************************************}    class function TOmfObjData.CodeSectionName(const aname: string): string;      begin{$ifdef i8086}        if current_settings.x86memorymodel in x86_far_code_models then          begin            if cs_huge_code in current_settings.moduleswitches then              result:=TrimStrCRC32(aname,30) + '_TEXT'            else              result:=current_module.modulename^ + '_TEXT';          end        else{$endif}          result:='_TEXT';      end;    constructor TOmfObjData.create(const n: string);      begin        inherited create(n);        CObjSymbol:=TOmfObjSymbol;        CObjSection:=TOmfObjSection;        createsectiongroup('DGROUP');        FMainSource:=current_module.mainsource;        FImportLibraryList:=TFPHashObjectList.Create(true);        FExportedSymbolList:=TFPHashObjectList.Create(true);      end;    destructor TOmfObjData.destroy;      begin        FExportedSymbolList.Free;        FImportLibraryList.Free;        inherited destroy;      end;    function TOmfObjData.sectiontype2align(atype: TAsmSectiontype): longint;      begin        Result:=omf_sectiontype2align(atype);      end;    class function TOmfObjData.sectiontype2class(atype: TAsmSectiontype): string;      begin        Result:=omf_segclass(atype);      end;    function TOmfObjData.sectionname(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder):string;      var        sep : string[3];        secname : string;      begin        if (atype=sec_user) then          Result:=aname        else          begin            if omf_secnames[atype]=omf_secnames[sec_code] then              secname:=CodeSectionName(aname)            else if omf_segclass(atype)='FAR_DATA' then              secname:=current_module.modulename^ + '_DATA'            else              secname:=omf_secnames[atype];            if create_smartlink_sections and (aname<>'') then              begin                case aorder of                  secorder_begin :                    sep:='.b_';                  secorder_end :                    sep:='.z_';                  else                    sep:='.n_';                end;                result:=UpCase(secname+sep+aname);              end            else              result:=secname;          end;      end;    function TOmfObjData.createsection(atype: TAsmSectionType; const aname: string; aorder: TAsmSectionOrder): TObjSection;      var        is_new: Boolean;        primary_group: String;        grp: TObjSectionGroup;      begin        is_new:=TObjSection(ObjSectionList.Find(sectionname(atype,aname,aorder)))=nil;        Result:=inherited createsection(atype, aname, aorder);        if is_new then          begin            TOmfObjSection(Result).FClassName:=sectiontype2class(atype);            if atype=sec_stack then              TOmfObjSection(Result).FCombination:=scStack            else if atype in [sec_debug_frame,sec_debug_info,sec_debug_line,sec_debug_abbrev,sec_debug_aranges,sec_debug_ranges] then              begin                TOmfObjSection(Result).FUse:=suUse32;                TOmfObjSection(Result).SizeLimit:=high(longword);              end;            primary_group:=omf_section_primary_group(atype,aname);            if primary_group<>'' then              begin                { find the primary group, if it already exists, else create it }                grp:=nil;                if GroupsList<>nil then                  grp:=TObjSectionGroup(GroupsList.Find(primary_group));                if grp=nil then                  grp:=createsectiongroup(primary_group);                { add the current section to the group }                SetLength(grp.members,Length(grp.members)+1);                grp.members[High(grp.members)]:=Result;                TOmfObjSection(Result).FPrimaryGroup:=grp;              end;          end;      end;    function TOmfObjData.reffardatasection: TObjSection;      var        secname: string;      begin        secname:=current_module.modulename^ + '_DATA';        result:=TObjSection(ObjSectionList.Find(secname));        if not assigned(result) then          begin            result:=CObjSection.create(ObjSectionList,secname,2,[oso_Data,oso_load,oso_write]);            result.ObjData:=self;            TOmfObjSection(Result).FClassName:='FAR_DATA';          end;      end;    procedure TOmfObjData.writeReloc(Data:TRelocDataInt;len:aword;p:TObjSymbol;Reloctype:TObjRelocationType);      var        objreloc: TOmfRelocation;        symaddr: AWord;      begin        { RELOC_FARPTR = RELOC_ABSOLUTE16+RELOC_SEG }        if Reloctype=RELOC_FARPTR then          begin            if len<>4 then              internalerror(2015041502);            writeReloc(Data,2,p,RELOC_ABSOLUTE16);            writeReloc(0,2,p,RELOC_SEG);            exit;          end        { RELOC_FARPTR48 = RELOC_ABSOLUTE16+RELOC_SEG }        else if Reloctype=RELOC_FARPTR48 then          begin            if len<>6 then              internalerror(2015041507);            writeReloc(Data,4,p,RELOC_ABSOLUTE32);            writeReloc(0,2,p,RELOC_SEG);            exit;          end;        if CurrObjSec=nil then          internalerror(2004030704);        objreloc:=nil;        if Reloctype in [RELOC_FARDATASEG,RELOC_FARDATASEGREL] then          begin            if Reloctype=RELOC_FARDATASEG then              objreloc:=TOmfRelocation.CreateSection(CurrObjSec.Size,reffardatasection,RELOC_SEG)            else              objreloc:=TOmfRelocation.CreateSection(CurrObjSec.Size,reffardatasection,RELOC_SEGREL);            CurrObjSec.ObjRelocations.Add(objreloc);          end        else if assigned(p) then          begin            { real address of the symbol }            symaddr:=p.address;            if p.bind=AB_EXTERNAL then              begin                objreloc:=TOmfRelocation.CreateSymbol(CurrObjSec.Size,p,Reloctype);                CurrObjSec.ObjRelocations.Add(objreloc);              end            { relative relocations within the same section can be calculated directly,              without the need to emit a relocation entry }            else if (p.objsection=CurrObjSec) and                    (p.bind<>AB_COMMON) and                    (Reloctype=RELOC_RELATIVE) then              begin                data:=data+symaddr-len-CurrObjSec.Size;              end            else              begin                objreloc:=TOmfRelocation.CreateSection(CurrObjSec.Size,p.objsection,Reloctype);                CurrObjSec.ObjRelocations.Add(objreloc);                if not (Reloctype in [RELOC_SEG,RELOC_SEGREL]) then                  inc(data,symaddr);              end;          end        else if Reloctype in [RELOC_DGROUP,RELOC_DGROUPREL] then            begin              if Reloctype=RELOC_DGROUP then                objreloc:=TOmfRelocation.CreateGroup(CurrObjSec.Size,TObjSectionGroup(GroupsList.Find('DGROUP')),RELOC_SEG)              else                objreloc:=TOmfRelocation.CreateGroup(CurrObjSec.Size,TObjSectionGroup(GroupsList.Find('DGROUP')),RELOC_SEGREL);              CurrObjSec.ObjRelocations.Add(objreloc);            end;        case len of          1:            CurrObjSec.write(data,1);          2:            CurrObjSec.writeInt16LE(int16(data));          4:            CurrObjSec.writeInt32LE(int32(data));          else            internalerror(2023110201);        end;      end;    procedure TOmfObjData.AddImportSymbol(const libname, symname,      symmangledname: TCmdStr; OrdNr: longint; isvar: boolean);      var        ImportLibrary : TImportLibrary;        ImportSymbol  : TFPHashObject;      begin        ImportLibrary:=TImportLibrary(ImportLibraryList.Find(libname));        if not assigned(ImportLibrary) then          ImportLibrary:=TImportLibrary.Create(ImportLibraryList,libname);        ImportSymbol:=TFPHashObject(ImportLibrary.ImportSymbolList.Find(symname));        if not assigned(ImportSymbol) then          ImportSymbol:=TImportSymbol.Create(ImportLibrary.ImportSymbolList,symname,symmangledname,OrdNr,isvar);      end;    procedure TOmfObjData.AddExportSymbol(aExportByOrdinal, aResidentName,        aNoData: Boolean; aParmCount: Integer; aExportedName,        aInternalName: string; aExportOrdinal: Word);      var        s: TOmfObjExportedSymbol;      begin        s:=TOmfObjExportedSymbol.Create(ExportedSymbolList,aInternalName);        with s do          begin            ExportByOrdinal:=aExportByOrdinal;            ResidentName:=aResidentName;            NoData:=aNoData;            ParmCount:=aParmCount;            ExportedName:=aExportedName;            InternalName:=aInternalName;            ExportOrdinal:=aExportOrdinal;          end;      end;{****************************************************************************                                TOmfObjOutput****************************************************************************}    procedure TOmfObjOutput.AddSegment(const name, segclass, ovlname: string;        Alignment: TOmfSegmentAlignment; Combination: TOmfSegmentCombination;        Use: TOmfSegmentUse; Size: TObjSectionOfs);      var        s: TOmfRecord_SEGDEF;      begin        s:=TOmfRecord_SEGDEF.Create;        Segments.Add(name,s);        s.SegmentNameIndex:=LNames.Add(name);        s.ClassNameIndex:=LNames.Add(segclass);        s.OverlayNameIndex:=LNames.Add(ovlname);        s.Alignment:=Alignment;        s.Combination:=Combination;        s.Use:=Use;        s.SegmentLength:=Size;      end;    procedure TOmfObjOutput.AddGroup(group: TObjSectionGroup);      var        g: TOmfRecord_GRPDEF;        seglist: TSegmentList;        I: Integer;      begin        seglist:=nil;        g:=TOmfRecord_GRPDEF.Create;        Groups.Add(group.Name,g);        g.GroupNameIndex:=LNames.Add(group.Name);        SetLength(seglist,Length(group.members));        for I:=Low(group.members) to High(group.members) do          seglist[I]:=group.members[I].index;        g.SegmentList:=seglist;      end;    procedure TOmfObjOutput.WriteSections(Data: TObjData);      var        i:longint;        sec:TObjSection;      begin        for i:=0 to Data.ObjSectionList.Count-1 do          begin            sec:=TObjSection(Data.ObjSectionList[i]);            WriteSectionContentAndFixups(sec);            WriteLinNumRecords(TOmfObjSection(sec));          end;      end;    procedure TOmfObjOutput.WriteSectionContentAndFixups(sec: TObjSection);      const        MaxChunkSize=$3fa;      var        RawRecord: TOmfRawRecord;        ChunkStart,ChunkLen: DWord;        ChunkFixupStart,ChunkFixupEnd: Integer;        SegIndex: Integer;        NextOfs: Integer;        Is32BitLEDATA: Boolean;        I: Integer;      begin        if (oso_data in sec.SecOptions) then          begin            if sec.Data=nil then              internalerror(2004030705);            for I:=0 to sec.ObjRelocations.Count-1 do              TOmfRelocation(sec.ObjRelocations[I]).BuildOmfFixup;            SegIndex:=Segments.FindIndexOf(sec.Name);            RawRecord:=TOmfRawRecord.Create;            sec.data.seek(0);            ChunkFixupStart:=0;            ChunkFixupEnd:=-1;            ChunkStart:=0;            ChunkLen:=Min(MaxChunkSize, sec.Data.size-ChunkStart);            while ChunkLen>0 do            begin              { find last fixup in the chunk }              while (ChunkFixupEnd<(sec.ObjRelocations.Count-1)) and                    (TOmfRelocation(sec.ObjRelocations[ChunkFixupEnd+1]).DataOffset<(ChunkStart+ChunkLen)) do                inc(ChunkFixupEnd);              { check if last chunk is crossing the chunk boundary, and trim ChunkLen if necessary }              if (ChunkFixupEnd>=ChunkFixupStart) and                ((TOmfRelocation(sec.ObjRelocations[ChunkFixupEnd]).DataOffset+                  TOmfRelocation(sec.ObjRelocations[ChunkFixupEnd]).OmfFixup.LocationSize)>(ChunkStart+ChunkLen)) then                begin                  ChunkLen:=TOmfRelocation(sec.ObjRelocations[ChunkFixupEnd]).DataOffset-ChunkStart;                  Dec(ChunkFixupEnd);                end;              { write LEDATA record }              Is32BitLEDATA:=TOmfObjSection(sec).Use=suUse32;              if Is32BitLEDATA then                RawRecord.RecordType:=RT_LEDATA32              else                RawRecord.RecordType:=RT_LEDATA;              NextOfs:=RawRecord.WriteIndexedRef(0,SegIndex);              if Is32BitLEDATA then                begin                  unaligned(PUint32(@RawRecord.RawData[NextOfs{..NextOfs+3}])^):=NtoLE(uint32(ChunkStart));                  Inc(NextOfs,4);                end              else                begin                  if ChunkStart>$ffff then                    internalerror(2018052201);                  unaligned(PUint16(@RawRecord.RawData[NextOfs{..NextOfs+1}])^):=NtoLE(uint16(ChunkStart));                  Inc(NextOfs,2);                end;              sec.data.read(RawRecord.RawData[NextOfs], ChunkLen);              Inc(NextOfs, ChunkLen);              RawRecord.RecordLength:=NextOfs+1;              RawRecord.CalculateChecksumByte;              RawRecord.WriteTo(FWriter);              { write FIXUPP record }              if ChunkFixupEnd>=ChunkFixupStart then                begin                  RawRecord.RecordType:=RT_FIXUPP;                  NextOfs:=0;                  for I:=ChunkFixupStart to ChunkFixupEnd do                    begin                      TOmfRelocation(sec.ObjRelocations[I]).OmfFixup.DataRecordStartOffset:=ChunkStart;                      NextOfs:=TOmfRelocation(sec.ObjRelocations[I]).OmfFixup.WriteAt(RawRecord,NextOfs);                    end;                  RawRecord.RecordLength:=NextOfs+1;                  RawRecord.CalculateChecksumByte;                  RawRecord.WriteTo(FWriter);                end;              { prepare next chunk }              Inc(ChunkStart, ChunkLen);              ChunkLen:=Min(MaxChunkSize, sec.Data.size-ChunkStart);              ChunkFixupStart:=ChunkFixupEnd+1;            end;            RawRecord.Free;          end;      end;    procedure TOmfObjOutput.WriteLinNumRecords(sec: TOmfObjSection);      var        SegIndex: Integer;        RawRecord: TOmfRawRecord;        LinNumRec: TOmfRecord_LINNUM_MsLink;      begin        if (oso_data in sec.SecOptions) then          begin            if sec.Data=nil then              internalerror(2004030706);            if sec.LinNumEntries.Count=0 then              exit;            SegIndex:=Segments.FindIndexOf(sec.Name);            RawRecord:=TOmfRawRecord.Create;            LinNumRec:=TOmfRecord_LINNUM_MsLink.Create;            LinNumRec.BaseGroup:=0;            LinNumRec.BaseSegment:=SegIndex;            LinNumRec.LineNumberList:=sec.LinNumEntries;            while LinNumRec.NextIndex<sec.LinNumEntries.Count do              begin                LinNumRec.EncodeTo(RawRecord);                RawRecord.WriteTo(FWriter);              end;            LinNumRec.Free;            RawRecord.Free;          end;      end;    procedure TOmfObjOutput.section_count_sections(p: TObject; arg: pointer);      begin        TOmfObjSection(p).index:=pinteger(arg)^;        inc(pinteger(arg)^);      end;    procedure TOmfObjOutput.group_count_groups(p: TObject; arg: pointer);      begin        TObjSectionGroup(p).index:=pinteger(arg)^;        inc(pinteger(arg)^);      end;    procedure TOmfObjOutput.WritePUBDEFs(Data: TObjData);      var        PubNamesForSection: array of TFPHashObjectList;        i: Integer;        objsym: TObjSymbol;        PublicNameElem: TOmfPublicNameElement;        RawRecord: TOmfRawRecord;        PubDefRec: TOmfRecord_PUBDEF;      begin        PubNamesForSection:=nil;        RawRecord:=TOmfRawRecord.Create;        SetLength(PubNamesForSection,Data.ObjSectionList.Count);        for i:=0 to Data.ObjSectionList.Count-1 do          PubNamesForSection[i]:=TFPHashObjectList.Create;        for i:=0 to Data.ObjSymbolList.Count-1 do          begin            objsym:=TObjSymbol(Data.ObjSymbolList[i]);            if objsym.bind=AB_GLOBAL then              begin                PublicNameElem:=TOmfPublicNameElement.Create(PubNamesForSection[objsym.objsection.index-1],objsym.Name);                PublicNameElem.PublicOffset:=objsym.offset;                PublicNameElem.IsLocal:=False;              end            else if objsym.bind=AB_LOCAL then              begin                PublicNameElem:=TOmfPublicNameElement.Create(PubNamesForSection[objsym.objsection.index-1],objsym.Name);                PublicNameElem.PublicOffset:=objsym.offset;                PublicNameElem.IsLocal:=True;              end          end;        for i:=0 to Data.ObjSectionList.Count-1 do          if PubNamesForSection[i].Count>0 then            begin              PubDefRec:=TOmfRecord_PUBDEF.Create;              PubDefRec.BaseSegmentIndex:=i+1;              if TOmfObjSection(Data.ObjSectionList[i]).PrimaryGroup<>nil then                PubDefRec.BaseGroupIndex:=Groups.FindIndexOf(TOmfObjSection(Data.ObjSectionList[i]).PrimaryGroup.Name)              else                PubDefRec.BaseGroupIndex:=0;              PubDefRec.PublicNames:=PubNamesForSection[i];              while PubDefRec.NextIndex<PubDefRec.PublicNames.Count do                begin                  PubDefRec.EncodeTo(RawRecord);                  RawRecord.WriteTo(FWriter);                end;              PubDefRec.Free;            end;        for i:=0 to Data.ObjSectionList.Count-1 do          FreeAndNil(PubNamesForSection[i]);        RawRecord.Free;      end;    procedure TOmfObjOutput.WriteEXTDEFs(Data: TObjData);      var        ExtNames: TFPHashObjectList;        RawRecord: TOmfRawRecord;        i,idx: Integer;        objsym: TObjSymbol;        ExtDefRec: TOmfRecord_EXTDEF;      begin        ExtNames:=TFPHashObjectList.Create;        RawRecord:=TOmfRawRecord.Create;        idx:=1;        for i:=0 to Data.ObjSymbolList.Count-1 do          begin            objsym:=TObjSymbol(Data.ObjSymbolList[i]);            if objsym.bind=AB_EXTERNAL then              begin                TOmfExternalNameElement.Create(ExtNames,objsym.Name);                objsym.symidx:=idx;                Inc(idx);              end;          end;        if ExtNames.Count>0 then          begin            ExtDefRec:=TOmfRecord_EXTDEF.Create;            ExtDefRec.ExternalNames:=ExtNames;            while ExtDefRec.NextIndex<ExtDefRec.ExternalNames.Count do              begin                ExtDefRec.EncodeTo(RawRecord);                RawRecord.WriteTo(FWriter);              end;            ExtDefRec.Free;          end;        ExtNames.Free;        RawRecord.Free;      end;    function TOmfObjOutput.writeData(Data:TObjData):boolean;      var        RawRecord: TOmfRawRecord;        Header: TOmfRecord_THEADR;        Translator_COMENT: TOmfRecord_COMENT;        DebugFormat_COMENT: TOmfRecord_COMENT;        LinkPassSeparator_COMENT: TOmfRecord_COMENT;        LNamesRec: TOmfRecord_LNAMES;        ModEnd: TOmfRecord_MODEND;        I: Integer;        SegDef: TOmfRecord_SEGDEF;        GrpDef: TOmfRecord_GRPDEF;        nsections,ngroups: Integer;      begin        { calc amount of sections we have and set their index, starting with 1 }        nsections:=1;        data.ObjSectionList.ForEachCall(@section_count_sections,@nsections);        { calc amount of groups we have and set their index, starting with 1 }        ngroups:=1;        data.GroupsList.ForEachCall(@group_count_groups,@ngroups);        { maximum amount of sections supported in the omf format is $7fff }        if (nsections-1)>$7fff then          internalerror(2015040701);        { maximum amount of groups supported in the omf format is $7fff }        if (ngroups-1)>$7fff then          internalerror(2018062101);        { write header record }        RawRecord:=TOmfRawRecord.Create;        Header:=TOmfRecord_THEADR.Create;        if cs_debuginfo in current_settings.moduleswitches then          Header.ModuleName:=TOmfObjData(Data).MainSource        else          Header.ModuleName:=Data.Name;        Header.EncodeTo(RawRecord);        RawRecord.WriteTo(FWriter);        Header.Free;        { write translator COMENT header }        Translator_COMENT:=TOmfRecord_COMENT.Create;        Translator_COMENT.CommentClass:=CC_Translator;        Translator_COMENT.CommentString:='FPC '+full_version_string+        ' ['+date_string+'] for '+target_cpu_string+' - '+target_info.shortname;        Translator_COMENT.EncodeTo(RawRecord);        RawRecord.WriteTo(FWriter);        Translator_COMENT.Free;        if (target_dbg.id=dbg_codeview) or           ((ds_dwarf_omf_linnum in current_settings.debugswitches) and            (target_dbg.id in [dbg_dwarf2,dbg_dwarf3,dbg_dwarf4])) then          begin            DebugFormat_COMENT:=TOmfRecord_COMENT.Create;            DebugFormat_COMENT.CommentClass:=CC_NewOmfExtension;            DebugFormat_COMENT.CommentString:='';            DebugFormat_COMENT.EncodeTo(RawRecord);            RawRecord.WriteTo(FWriter);            DebugFormat_COMENT.Free;          end;        LNames.Clear;        LNames.Add('');  { insert an empty string, which has index 1 }        FSegments.Clear;        FSegments.Add('',nil);        FGroups.Clear;        FGroups.Add('',nil);        for i:=0 to Data.GroupsList.Count-1 do          AddGroup(TObjSectionGroup(Data.GroupsList[I]));        for i:=0 to Data.ObjSectionList.Count-1 do          with TOmfObjSection(Data.ObjSectionList[I]) do            AddSegment(Name,ClassName,OverlayName,OmfAlignment,Combination,Use,Size);        { write LNAMES record(s) }        LNamesRec:=TOmfRecord_LNAMES.Create;        LNamesRec.Names:=LNames;        while LNamesRec.NextIndex<=LNames.Count do          begin            LNamesRec.EncodeTo(RawRecord);            RawRecord.WriteTo(FWriter);          end;        LNamesRec.Free;        { write SEGDEF record(s) }        for I:=1 to Segments.Count-1 do          begin            SegDef:=TOmfRecord_SEGDEF(Segments[I]);            SegDef.EncodeTo(RawRecord);            RawRecord.WriteTo(FWriter);          end;        { write GRPDEF record(s) }        for I:=1 to Groups.Count-1 do          begin            GrpDef:=TOmfRecord_GRPDEF(Groups[I]);            GrpDef.EncodeTo(RawRecord);            RawRecord.WriteTo(FWriter);          end;        { write PUBDEF record(s) }        WritePUBDEFs(Data);        { write EXTDEF record(s) }        WriteEXTDEFs(Data);        { write link pass separator }        LinkPassSeparator_COMENT:=TOmfRecord_COMENT.Create;        LinkPassSeparator_COMENT.CommentClass:=CC_LinkPassSeparator;        LinkPassSeparator_COMENT.CommentString:=#1;        LinkPassSeparator_COMENT.NoList:=True;        LinkPassSeparator_COMENT.EncodeTo(RawRecord);        RawRecord.WriteTo(FWriter);        LinkPassSeparator_COMENT.Free;        { write section content, interleaved with fixups }        WriteSections(Data);        { write MODEND record }        ModEnd:=TOmfRecord_MODEND.Create;        ModEnd.EncodeTo(RawRecord);        RawRecord.WriteTo(FWriter);        ModEnd.Free;        RawRecord.Free;        result:=true;      end;    constructor TOmfObjOutput.create(AWriter:TObjectWriter);      begin        inherited create(AWriter);        cobjdata:=TOmfObjData;        FLNames:=TOmfOrderedNameCollection.Create(False);        FSegments:=TFPHashObjectList.Create;        FSegments.Add('',nil);        FGroups:=TFPHashObjectList.Create;        FGroups.Add('',nil);      end;    destructor TOmfObjOutput.Destroy;      begin        FGroups.Free;        FSegments.Free;        FLNames.Free;        inherited Destroy;      end;    procedure TOmfObjOutput.WriteDllImport(const dllname,afuncname,mangledname: string; ordnr: longint; isvar: boolean);      var        RawRecord: TOmfRawRecord;        Header: TOmfRecord_THEADR;        DllImport_COMENT: TOmfRecord_COMENT=nil;        DllImport_COMENT_IMPDEF: TOmfRecord_COMENT_IMPDEF=nil;        ModEnd: TOmfRecord_MODEND;      begin        { write header record }        RawRecord:=TOmfRawRecord.Create;        Header:=TOmfRecord_THEADR.Create;        Header.ModuleName:=mangledname;        Header.EncodeTo(RawRecord);        RawRecord.WriteTo(FWriter);        Header.Free;        { write IMPDEF record }        DllImport_COMENT_IMPDEF:=TOmfRecord_COMENT_IMPDEF.Create;        DllImport_COMENT_IMPDEF.InternalName:=mangledname;        DllImport_COMENT_IMPDEF.ModuleName:=dllname;        if ordnr <= 0 then          begin            DllImport_COMENT_IMPDEF.ImportByOrdinal:=False;            DllImport_COMENT_IMPDEF.Name:=afuncname;          end        else          begin            DllImport_COMENT_IMPDEF.ImportByOrdinal:=True;            DllImport_COMENT_IMPDEF.Ordinal:=ordnr;          end;        DllImport_COMENT:=TOmfRecord_COMENT.Create;        DllImport_COMENT_IMPDEF.EncodeTo(DllImport_COMENT);        FreeAndNil(DllImport_COMENT_IMPDEF);        DllImport_COMENT.EncodeTo(RawRecord);        FreeAndNil(DllImport_COMENT);        RawRecord.WriteTo(FWriter);        { write MODEND record }        ModEnd:=TOmfRecord_MODEND.Create;        ModEnd.EncodeTo(RawRecord);        RawRecord.WriteTo(FWriter);        ModEnd.Free;        RawRecord.Free;      end;{****************************************************************************                               TOmfObjInput****************************************************************************}    function TOmfObjInput.PeekNextRecordType: Byte;      var        OldPos: LongInt;      begin        OldPos:=FReader.Pos;        if not FReader.read(Result, 1) then          begin            InputError('Unexpected end of file');            Result:=0;            exit;          end;        FReader.seek(OldPos);      end;    function TOmfObjInput.ReadLNames(RawRec: TOmfRawRecord): Boolean;      var        LNamesRec: TOmfRecord_LNAMES;      begin        Result:=False;        LNamesRec:=TOmfRecord_LNAMES.Create;        LNamesRec.Names:=LNames;        LNamesRec.DecodeFrom(RawRec);        LNamesRec.Free;        Result:=True;      end;    function TOmfObjInput.ReadSegDef(RawRec: TOmfRawRecord; objdata: TObjData): Boolean;      var        SegDefRec: TOmfRecord_SEGDEF;        SegmentName,SegClassName,OverlayName: string;        SecAlign: LongInt;        secoptions: TObjSectionOptions;        objsec: TOmfObjSection;      begin        Result:=False;        SegDefRec:=TOmfRecord_SEGDEF.Create;        SegDefRec.DecodeFrom(RawRec);        if (SegDefRec.SegmentNameIndex<1) or (SegDefRec.SegmentNameIndex>LNames.Count) then          begin            InputError('Segment name index out of range');            SegDefRec.Free;            exit;          end;        SegmentName:=LNames[SegDefRec.SegmentNameIndex];        if (SegDefRec.ClassNameIndex<1) or (SegDefRec.ClassNameIndex>LNames.Count) then          begin            InputError('Segment class name index out of range');            SegDefRec.Free;            exit;          end;        SegClassName:=LNames[SegDefRec.ClassNameIndex];        if (SegDefRec.OverlayNameIndex<1) or (SegDefRec.OverlayNameIndex>LNames.Count) then          begin            InputError('Segment overlay name index out of range');            SegDefRec.Free;            exit;          end;        OverlayName:=LNames[SegDefRec.OverlayNameIndex];        SecAlign:=1; // otherwise warning prohibits compilation        case SegDefRec.Alignment of          saRelocatableByteAligned:            SecAlign:=1;          saRelocatableWordAligned:            SecAlign:=2;          saRelocatableParaAligned:            SecAlign:=16;          saRelocatableDWordAligned:            SecAlign:=4;          saRelocatablePageAligned:            SecAlign:=256;          saNotSupported:            SecAlign:=4096;          saAbsolute:            begin              InputError('Absolute segment alignment not supported');              SegDefRec.Free;              exit;            end;          saNotDefined:            begin              InputError('Invalid (unsupported/undefined) OMF segment alignment');              SegDefRec.Free;              exit;            end;        end;        if not CaseSensitiveSegments then          begin            SegmentName:=UpCase(SegmentName);            SegClassName:=UpCase(SegClassName);            OverlayName:=UpCase(OverlayName);          end;        { hack for supporting object modules, generated by Borland's BINOBJ tool }        if (SegClassName='') and (SegmentName='CODE') then          begin            SegmentName:=InputFileName;            SegClassName:='CODE';          end;        secoptions:=[];        objsec:=TOmfObjSection(objdata.createsection(SegmentName+'||'+SegClassName,SecAlign,secoptions,false));        objsec.FClassName:=SegClassName;        objsec.FOverlayName:=OverlayName;        objsec.FCombination:=SegDefRec.Combination;        objsec.FUse:=SegDefRec.Use;        if SegDefRec.SegmentLength>High(objsec.Size) then          begin            InputError('Segment too large');            SegDefRec.Free;            exit;          end;        objsec.Size:=SegDefRec.SegmentLength;        if SegClassName='DWARF' then          objsec.SecOptions:=objsec.SecOptions+[oso_debug];        if (SegClassName='HEAP') or           (SegClassName='STACK') or (SegDefRec.Combination=scStack) or           (SegClassName='BEGDATA') or           (SegmentName='FPC') then          objsec.SecOptions:=objsec.SecOptions+[oso_keep];        SegDefRec.Free;        Result:=True;      end;    function TOmfObjInput.ReadGrpDef(RawRec: TOmfRawRecord; objdata: TObjData): Boolean;      var        GrpDefRec: TOmfRecord_GRPDEF;        GroupName: string;        SecGroup: TObjSectionGroup;        i,SegIndex: Integer;      begin        Result:=False;        GrpDefRec:=TOmfRecord_GRPDEF.Create;        GrpDefRec.DecodeFrom(RawRec);        if (GrpDefRec.GroupNameIndex<1) or (GrpDefRec.GroupNameIndex>LNames.Count) then          begin            InputError('Group name index out of range');            GrpDefRec.Free;            exit;          end;        GroupName:=LNames[GrpDefRec.GroupNameIndex];        if not CaseSensitiveSegments then          GroupName:=UpCase(GroupName);        SecGroup:=objdata.createsectiongroup(GroupName);        SetLength(SecGroup.members,Length(GrpDefRec.SegmentList));        for i:=0 to Length(GrpDefRec.SegmentList)-1 do          begin            SegIndex:=GrpDefRec.SegmentList[i];            if (SegIndex<1) or (SegIndex>objdata.ObjSectionList.Count) then              begin                InputError('Segment name index out of range in group definition');                GrpDefRec.Free;                exit;              end;            SecGroup.members[i]:=TOmfObjSection(objdata.ObjSectionList[SegIndex-1]);          end;        GrpDefRec.Free;        Result:=True;      end;    function TOmfObjInput.ReadExtDef(RawRec: TOmfRawRecord; objdata: TObjData): Boolean;      var        ExtDefRec: TOmfRecord_EXTDEF;        ExtDefElem: TOmfExternalNameElement;        OldCount,NewCount,i: Integer;        objsym: TObjSymbol;        symname: TSymStr;      begin        Result:=False;        ExtDefRec:=TOmfRecord_EXTDEF.Create;        ExtDefRec.ExternalNames:=ExtDefs;        OldCount:=ExtDefs.Count;        ExtDefRec.DecodeFrom(RawRec);        NewCount:=ExtDefs.Count;        for i:=OldCount to NewCount-1 do          begin            ExtDefElem:=TOmfExternalNameElement(ExtDefs[i]);            symname:=ExtDefElem.Name;            if not CaseSensitiveSymbols then              symname:=UpCase(symname);            objsym:=objdata.CreateSymbol(symname);            objsym.bind:=AB_EXTERNAL;            objsym.typ:=AT_FUNCTION;            objsym.objsection:=nil;            objsym.offset:=0;            objsym.size:=0;          end;        ExtDefRec.Free;        Result:=True;      end;    function TOmfObjInput.ReadPubDef(RawRec: TOmfRawRecord; objdata:TObjData): Boolean;      var        PubDefRec: TOmfRecord_PUBDEF;        PubDefElem: TOmfPublicNameElement;        OldCount,NewCount,i: Integer;        basegroup: TObjSectionGroup;        objsym: TObjSymbol;        objsec: TOmfObjSection;        symname: TSymStr;      begin        Result:=False;        PubDefRec:=TOmfRecord_PUBDEF.Create;        PubDefRec.PublicNames:=PubDefs;        OldCount:=PubDefs.Count;        PubDefRec.DecodeFrom(RawRec);        NewCount:=PubDefs.Count;        if (PubDefRec.BaseGroupIndex<0) or (PubDefRec.BaseGroupIndex>objdata.GroupsList.Count) then          begin            InputError('Public symbol''s group name index out of range');            PubDefRec.Free;            exit;          end;        if PubDefRec.BaseGroupIndex<>0 then          basegroup:=TObjSectionGroup(objdata.GroupsList[PubDefRec.BaseGroupIndex-1])        else          basegroup:=nil;        if (PubDefRec.BaseSegmentIndex<0) or (PubDefRec.BaseSegmentIndex>objdata.ObjSectionList.Count) then          begin            InputError('Public symbol''s segment name index out of range');            PubDefRec.Free;            exit;          end;        if PubDefRec.BaseSegmentIndex=0 then          begin            InputError('Public symbol uses absolute addressing, which is not supported by this linker');            PubDefRec.Free;            exit;          end;        objsec:=TOmfObjSection(objdata.ObjSectionList[PubDefRec.BaseSegmentIndex-1]);        for i:=OldCount to NewCount-1 do          begin            PubDefElem:=TOmfPublicNameElement(PubDefs[i]);            symname:=PubDefElem.Name;            if not CaseSensitiveSymbols then              symname:=UpCase(symname);            objsym:=objdata.CreateSymbol(symname);            if PubDefElem.IsLocal then              objsym.bind:=AB_LOCAL            else              objsym.bind:=AB_GLOBAL;            objsym.typ:=AT_FUNCTION;            objsym.group:=basegroup;            objsym.objsection:=objsec;            objsym.offset:=PubDefElem.PublicOffset;            objsym.size:=0;            if (objsym.bind=AB_GLOBAL) and (objsec.FFirstSym=nil) then              objsec.FFirstSym:=objsym;          end;        PubDefRec.Free;        Result:=True;      end;    function TOmfObjInput.ReadModEnd(RawRec: TOmfRawRecord; objdata:TObjData): Boolean;      var        ModEndRec: TOmfRecord_MODEND;        objsym: TObjSymbol;        objsec: TOmfObjSection;        basegroup: TObjSectionGroup;      begin        Result:=False;        ModEndRec:=TOmfRecord_MODEND.Create;        ModEndRec.DecodeFrom(RawRec);        if ModEndRec.HasStartAddress then          begin            if not ModEndRec.LogicalStartAddress then              begin                InputError('Physical start address not supported');                ModEndRec.Free;                exit;              end;            if not (ModEndRec.TargetMethod in [ftmSegmentIndex,ftmSegmentIndexNoDisp]) then              begin                InputError('Target method for start address other than "Segment Index" is not supported');                ModEndRec.Free;                exit;              end;            if (ModEndRec.TargetDatum<1) or (ModEndRec.TargetDatum>objdata.ObjSectionList.Count) then              begin                InputError('Segment name index for start address out of range');                ModEndRec.Free;                exit;              end;            case ModEndRec.FrameMethod of              ffmSegmentIndex:                begin                  if (ModEndRec.FrameDatum<1) or (ModEndRec.FrameDatum>objdata.ObjSectionList.Count) then                    begin                      InputError('Frame segment name index for start address out of range');                      ModEndRec.Free;                      exit;                    end;                  if ModEndRec.FrameDatum<>ModEndRec.TargetDatum then                    begin                      InputError('Frame segment different than target segment is not supported supported for start address');                      ModEndRec.Free;                      exit;                    end;                  basegroup:=nil;                end;              ffmGroupIndex:                begin                  if (ModEndRec.FrameDatum<1) or (ModEndRec.FrameDatum>objdata.GroupsList.Count) then                    begin                      InputError('Frame group name index for start address out of range');                      ModEndRec.Free;                      exit;                    end;                  basegroup:=TObjSectionGroup(objdata.GroupsList[ModEndRec.FrameDatum-1]);                end;              else                begin                  InputError('Frame method for start address other than "Segment Index" or "Group Index" is not supported');                  ModEndRec.Free;                  exit;                end;            end;            objsec:=TOmfObjSection(objdata.ObjSectionList[ModEndRec.TargetDatum-1]);            objsym:=objdata.CreateSymbol('..start');            objsym.bind:=AB_GLOBAL;            objsym.typ:=AT_FUNCTION;            objsym.group:=basegroup;            objsym.objsection:=objsec;            objsym.offset:=ModEndRec.TargetDisplacement;            objsym.size:=0;          end;        ModEndRec.Free;        Result:=True;      end;    function TOmfObjInput.ReadLeOrLiDataAndFixups(RawRec: TOmfRawRecord; objdata: TObjData): Boolean;      var        Is32Bit: Boolean;        NextOfs: Integer;        SegmentIndex: Integer;        EnumeratedDataOffset: DWord;        BlockLength: Integer;        objsec: TOmfObjSection;        FixupRawRec: TOmfRawRecord=nil;        Fixup: TOmfSubRecord_FIXUP;        Thread: TOmfSubRecord_THREAD;        FixuppWithoutLeOrLiData: Boolean=False;      begin        objsec:=nil;        EnumeratedDataOffset:=0;        Result:=False;        case RawRec.RecordType of          RT_LEDATA,RT_LEDATA32:            begin              Is32Bit:=RawRec.RecordType=RT_LEDATA32;              NextOfs:=RawRec.ReadIndexedRef(0,SegmentIndex);              if Is32Bit then                begin                  if (NextOfs+3)>=RawRec.RecordLength then                    internalerror(2015040512);                  EnumeratedDataOffset := LEtoN(unaligned(PUint32(@RawRec.RawData[NextOfs{..NextOfs+3}])^));                  Inc(NextOfs,4);                end              else                begin                  if (NextOfs+1)>=RawRec.RecordLength then                    internalerror(2015040513);                  EnumeratedDataOffset := LEtoN(unaligned(PUint16(@RawRec.RawData[NextOfs{..NextOfs+1}])^));                  Inc(NextOfs,2);                end;              BlockLength:=RawRec.RecordLength-NextOfs-1;              if BlockLength<0 then                internalerror(2015060501);              if BlockLength>1024 then                begin                  InputError('LEDATA contains more than 1024 bytes of data');                  exit;                end;              if (SegmentIndex<1) or (SegmentIndex>objdata.ObjSectionList.Count) then                begin                  InputError('Segment index in LEDATA field is out of range');                  exit;                end;              objsec:=TOmfObjSection(objdata.ObjSectionList[SegmentIndex-1]);              objsec.SecOptions:=objsec.SecOptions+[oso_Data];              if (objsec.Data.Size>EnumeratedDataOffset) then                begin                  InputError('LEDATA enumerated data offset field out of sequence');                  exit;                end;              if (EnumeratedDataOffset+BlockLength)>objsec.Size then                begin                  InputError('LEDATA goes beyond the segment size declared in the SEGDEF record');                  exit;                end;              objsec.Data.seek(EnumeratedDataOffset);              objsec.Data.write(RawRec.RawData[NextOfs],BlockLength);            end;          RT_LIDATA,RT_LIDATA32:            begin              InputError('LIDATA records are not supported');              exit;            end;          RT_FIXUPP,RT_FIXUPP32:            begin              FixuppWithoutLeOrLiData:=True;              { a hack, used to indicate, that we must process this record       }              { (RawRec) first in the FIXUPP record processing loop that follows }              FixupRawRec:=RawRec;            end;          else            internalerror(2015040316);        end;        { also read all the FIXUPP records that may follow;                     }        { (FixupRawRec=RawRec) indicates that we must process RawRec first, but }        { without freeing it                                                    }        while (FixupRawRec=RawRec) or (PeekNextRecordType in [RT_FIXUPP,RT_FIXUPP32]) do          begin            if FixupRawRec<>RawRec then              begin                FixupRawRec:=TOmfRawRecord.Create;                FixupRawRec.ReadFrom(FReader);                if not FRawRecord.VerifyChecksumByte then                  begin                    InputError('Invalid checksum in OMF record');                    FixupRawRec.Free;                    exit;                  end;              end;            NextOfs:=0;            Thread:=TOmfSubRecord_THREAD.Create;            Fixup:=TOmfSubRecord_FIXUP.Create;            Fixup.Is32Bit:=FixupRawRec.RecordType=RT_FIXUPP32;            Fixup.DataRecordStartOffset:=EnumeratedDataOffset;            while NextOfs<(FixupRawRec.RecordLength-1) do              begin                if (FixupRawRec.RawData[NextOfs] and $80)<>0 then                  begin                    { FIXUP subrecord }                    if FixuppWithoutLeOrLiData then                      begin                        InputError('FIXUP subrecord without previous LEDATA or LIDATA record');                        Fixup.Free;                        Thread.Free;                        if FixupRawRec<>RawRec then                          FixupRawRec.Free;                        exit;                      end;                    NextOfs:=Fixup.ReadAt(FixupRawRec,NextOfs);                    Fixup.ResolveByThread(FFixupThreads);                    ImportOmfFixup(objdata,objsec,Fixup);                  end                else                  begin                    { THREAD subrecord }                    NextOfs:=Thread.ReadAt(FixupRawRec,NextOfs);                    Thread.ApplyTo(FFixupThreads);                  end;              end;            Fixup.Free;            Thread.Free;            if FixupRawRec<>RawRec then              FixupRawRec.Free;            { always set it to null, so that we read the next record on the next }            { loop iteration (this ensures that FixupRawRec<>RawRec, without     }            { freeing RawRec)                                                    }            FixupRawRec:=nil;          end;        Result:=True;      end;    function TOmfObjInput.ReadImpDef(Rec: TOmfRecord_COMENT; objdata: TObjData): Boolean;      var        ImpDefRec: TOmfRecord_COMENT_IMPDEF;        SymName: string;      begin        ImpDefRec:=TOmfRecord_COMENT_IMPDEF.Create;        ImpDefRec.DecodeFrom(Rec);        SymName:=ImpDefRec.InternalName;        if not CaseSensitiveSymbols then          SymName:=UpCase(SymName);        if ImpDefRec.ImportByOrdinal then          TOmfObjData(objdata).AddImportSymbol(MaybeAddDllExt(ImpDefRec.ModuleName),'',SymName,ImpDefRec.Ordinal,false)        else          TOmfObjData(objdata).AddImportSymbol(MaybeAddDllExt(ImpDefRec.ModuleName),ImpDefRec.Name,SymName,0,false);        Result:=True;        ImpDefRec.Free;      end;    function TOmfObjInput.ReadExpDef(Rec: TOmfRecord_COMENT; objdata: TObjData): Boolean;      var        ExpDefRec: TOmfRecord_COMENT_EXPDEF;        SymName: string;      begin        ExpDefRec:=TOmfRecord_COMENT_EXPDEF.Create;        ExpDefRec.DecodeFrom(Rec);        SymName:=ExpDefRec.InternalName;        if not CaseSensitiveSymbols then          SymName:=UpCase(SymName);        TOmfObjData(objdata).AddExportSymbol(          ExpDefRec.ExportByOrdinal,          ExpDefRec.ResidentName,          ExpDefRec.NoData,          ExpDefRec.ParmCount,          ExpDefRec.ExportedName,          SymName,          ExpDefRec.ExportOrdinal);        Result:=True;        ExpDefRec.Free;      end;    function TOmfObjInput.ImportOmfFixup(objdata: TObjData; objsec: TOmfObjSection; Fixup: TOmfSubRecord_FIXUP): Boolean;      var        reloc: TOmfRelocation;        sym: TObjSymbol;        RelocType: TObjRelocationType;        target_section: TOmfObjSection;        target_group: TObjSectionGroup;      begin        Result:=False;        { range check location }        if (Fixup.LocationOffset+Fixup.LocationSize)>objsec.Size then          begin            InputError('Fixup location exceeds the current segment boundary');            exit;          end;        { range check target datum }        case Fixup.TargetMethod of          ftmSegmentIndex:            if (Fixup.TargetDatum<1) or (Fixup.TargetDatum>objdata.ObjSectionList.Count) then              begin                InputError('Segment name index in SI(<segment name>),<displacement> fixup target is out of range');                exit;              end;          ftmSegmentIndexNoDisp:            if (Fixup.TargetDatum<1) or (Fixup.TargetDatum>objdata.ObjSectionList.Count) then              begin                InputError('Segment name index in SI(<segment name>) fixup target is out of range');                exit;              end;          ftmGroupIndex:            if (Fixup.TargetDatum<1) or (Fixup.TargetDatum>objdata.GroupsList.Count) then              begin                InputError('Group name index in GI(<group name>),<displacement> fixup target is out of range');                exit;              end;          ftmGroupIndexNoDisp:            if (Fixup.TargetDatum<1) or (Fixup.TargetDatum>objdata.GroupsList.Count) then              begin                InputError('Group name index in GI(<group name>) fixup target is out of range');                exit;              end;          ftmExternalIndex:            if (Fixup.TargetDatum<1) or (Fixup.TargetDatum>ExtDefs.Count) then              begin                InputError('External symbol name index in EI(<symbol name>),<displacement> fixup target is out of range');                exit;              end;          ftmExternalIndexNoDisp:            begin              if (Fixup.TargetDatum<1) or (Fixup.TargetDatum>ExtDefs.Count) then                begin                  InputError('External symbol name index in EI(<symbol name>) fixup target is out of range');                  exit;                end;            end;          else            ;        end;        { range check frame datum }        case Fixup.FrameMethod of          ffmSegmentIndex:            if (Fixup.FrameDatum<1) or (Fixup.FrameDatum>objdata.ObjSectionList.Count) then              begin                InputError('Segment name index in SI(<segment name>) fixup frame is out of range');                exit;              end;          ffmGroupIndex:            if (Fixup.FrameDatum<1) or (Fixup.FrameDatum>objdata.GroupsList.Count) then              begin                InputError('Group name index in GI(<group name>) fixup frame is out of range');                exit;              end;          ffmExternalIndex:            if (Fixup.TargetDatum<1) or (Fixup.TargetDatum>ExtDefs.Count) then              begin                InputError('External symbol name index in EI(<symbol name>) fixup frame is out of range');                exit;              end;          else            ;        end;        if Fixup.TargetMethod in [ftmExternalIndex,ftmExternalIndexNoDisp] then          begin            sym:=objdata.symbolref(TOmfExternalNameElement(ExtDefs[Fixup.TargetDatum-1]).Name);            RelocType:=RELOC_NONE;            case Fixup.LocationType of              fltOffset:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_ABSOLUTE16;                  fmSelfRelative:                    RelocType:=RELOC_RELATIVE16;                end;              fltOffset32:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_ABSOLUTE32;                  fmSelfRelative:                    RelocType:=RELOC_RELATIVE32;                end;              fltBase:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_SEG;                  fmSelfRelative:                    RelocType:=RELOC_SEGREL;                end;              fltFarPointer:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_FARPTR;                  fmSelfRelative:                    RelocType:=RELOC_FARPTR_RELATIVEOFFSET;                end;              fltFarPointer48:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_FARPTR48;                  fmSelfRelative:                    RelocType:=RELOC_FARPTR48_RELATIVEOFFSET;                end;              else                ;            end;            if RelocType=RELOC_NONE then              begin                InputError('Unsupported fixup location type '+tostr(Ord(Fixup.LocationType))+' with mode '+tostr(ord(Fixup.Mode))+' in external reference to '+sym.Name);                exit;              end;            reloc:=TOmfRelocation.CreateSymbol(Fixup.LocationOffset,sym,RelocType);            objsec.ObjRelocations.Add(reloc);            case Fixup.FrameMethod of              ffmTarget:                {nothing};              ffmGroupIndex:                reloc.FrameGroup:=TObjSectionGroup(objdata.GroupsList[Fixup.FrameDatum-1]).Name;              else                begin                  InputError('Unsupported frame method '+IntToStr(Ord(Fixup.FrameMethod))+' in external reference to '+sym.Name);                  exit;                end;            end;            if Fixup.TargetDisplacement<>0 then              begin                InputError('Unsupported nonzero target displacement '+IntToStr(Fixup.TargetDisplacement)+' in external reference to '+sym.Name);                exit;              end;          end        else if Fixup.TargetMethod in [ftmSegmentIndex,ftmSegmentIndexNoDisp] then          begin            target_section:=TOmfObjSection(objdata.ObjSectionList[Fixup.TargetDatum-1]);            RelocType:=RELOC_NONE;            case Fixup.LocationType of              fltOffset:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_ABSOLUTE16;                  fmSelfRelative:                    RelocType:=RELOC_RELATIVE16;                end;              fltOffset32:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_ABSOLUTE32;                  fmSelfRelative:                    RelocType:=RELOC_RELATIVE32;                end;              fltBase:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_SEG;                  fmSelfRelative:                    RelocType:=RELOC_SEGREL;                end;              fltFarPointer:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_FARPTR;                  fmSelfRelative:                    RelocType:=RELOC_FARPTR_RELATIVEOFFSET;                end;              fltFarPointer48:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_FARPTR48;                  fmSelfRelative:                    RelocType:=RELOC_FARPTR48_RELATIVEOFFSET;                end;              else                ;            end;            if RelocType=RELOC_NONE then              begin                InputError('Unsupported fixup location type '+tostr(Ord(Fixup.LocationType))+' with mode '+tostr(ord(Fixup.Mode)));                exit;              end;            reloc:=TOmfRelocation.CreateSection(Fixup.LocationOffset,target_section,RelocType);            objsec.ObjRelocations.Add(reloc);            case Fixup.FrameMethod of              ffmTarget:                {nothing};              ffmGroupIndex:                reloc.FrameGroup:=TObjSectionGroup(objdata.GroupsList[Fixup.FrameDatum-1]).Name;              else                begin                  InputError('Unsupported frame method '+IntToStr(Ord(Fixup.FrameMethod))+' in reference to segment '+target_section.Name);                  exit;                end;            end;            if Fixup.TargetDisplacement<>0 then              begin                InputError('Unsupported nonzero target displacement '+IntToStr(Fixup.TargetDisplacement)+' in reference to segment '+target_section.Name);                exit;              end;          end        else if Fixup.TargetMethod in [ftmGroupIndex,ftmGroupIndexNoDisp] then          begin            target_group:=TObjSectionGroup(objdata.GroupsList[Fixup.TargetDatum-1]);            RelocType:=RELOC_NONE;            case Fixup.LocationType of              fltOffset:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_ABSOLUTE16;                  fmSelfRelative:                    RelocType:=RELOC_RELATIVE16;                end;              fltOffset32:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_ABSOLUTE32;                  fmSelfRelative:                    RelocType:=RELOC_RELATIVE32;                end;              fltBase:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_SEG;                  fmSelfRelative:                    RelocType:=RELOC_SEGREL;                end;              fltFarPointer:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_FARPTR;                  fmSelfRelative:                    RelocType:=RELOC_FARPTR_RELATIVEOFFSET;                end;              fltFarPointer48:                case Fixup.Mode of                  fmSegmentRelative:                    RelocType:=RELOC_FARPTR48;                  fmSelfRelative:                    RelocType:=RELOC_FARPTR48_RELATIVEOFFSET;                end;              else                ;            end;            if RelocType=RELOC_NONE then              begin                InputError('Unsupported fixup location type '+tostr(Ord(Fixup.LocationType))+' with mode '+tostr(ord(Fixup.Mode)));                exit;              end;            reloc:=TOmfRelocation.CreateGroup(Fixup.LocationOffset,target_group,RelocType);            objsec.ObjRelocations.Add(reloc);            case Fixup.FrameMethod of              ffmTarget:                {nothing};              else                begin                  InputError('Unsupported frame method '+IntToStr(Ord(Fixup.FrameMethod))+' in reference to group '+target_group.Name);                  exit;                end;            end;            if Fixup.TargetDisplacement<>0 then              begin                InputError('Unsupported nonzero target displacement '+IntToStr(Fixup.TargetDisplacement)+' in reference to group '+target_group.Name);                exit;              end;          end        else          begin            {todo: convert other fixup types as well }            InputError('Unsupported fixup target method '+IntToStr(Ord(Fixup.TargetMethod)));            exit;          end;        Result:=True;      end;    constructor TOmfObjInput.create;      begin        inherited create;        cobjdata:=TOmfObjData;        FLNames:=TOmfOrderedNameCollection.Create(True);        FExtDefs:=TFPHashObjectList.Create;        FPubDefs:=TFPHashObjectList.Create;        FFixupThreads:=TOmfThreads.Create;        FRawRecord:=TOmfRawRecord.Create;        CaseSensitiveSegments:=False;        CaseSensitiveSymbols:=True;      end;    destructor TOmfObjInput.destroy;      begin        FCOMENTRecord.Free;        FRawRecord.Free;        FFixupThreads.Free;        FPubDefs.Free;        FExtDefs.Free;        FLNames.Free;        inherited destroy;      end;    class function TOmfObjInput.CanReadObjData(AReader: TObjectreader): boolean;      var        b: Byte;      begin        result:=false;        if AReader.Read(b,sizeof(b)) then          begin            if b=RT_THEADR then            { TODO: check additional fields }              result:=true;          end;        AReader.Seek(0);      end;    function TOmfObjInput.ReadObjData(AReader: TObjectreader; out objdata: TObjData): boolean;      begin        FReader:=AReader;        InputFileName:=AReader.FileName;        objdata:=CObjData.Create(InputFileName);        result:=false;        { the TOmfObjData constructor creates a group 'DGROUP', which is to be          used by the code generator, when writing files. When reading object          files, however, we need to start with an empty list of groups, so          let's clear the group list now. }        objdata.GroupsList.Clear;        LNames.Clear;        ExtDefs.Clear;        FRawRecord.ReadFrom(FReader);        if not FRawRecord.VerifyChecksumByte then          begin            InputError('Invalid checksum in OMF record');            exit;          end;        if FRawRecord.RecordType<>RT_THEADR then          begin            InputError('Can''t read OMF header');            exit;          end;        repeat          FRawRecord.ReadFrom(FReader);          if not FRawRecord.VerifyChecksumByte then            begin              InputError('Invalid checksum in OMF record');              exit;            end;          FreeAndNil(FCOMENTRecord);          case FRawRecord.RecordType of            RT_LNAMES:              if not ReadLNames(FRawRecord) then                exit;            RT_SEGDEF,RT_SEGDEF32:              if not ReadSegDef(FRawRecord,objdata) then                exit;            RT_GRPDEF:              if not ReadGrpDef(FRawRecord,objdata) then                exit;            RT_COMENT:              begin                FCOMENTRecord:=TOmfRecord_COMENT.Create;                FCOMENTRecord.DecodeFrom(FRawRecord);                case FCOMENTRecord.CommentClass of                  CC_OmfExtension:                    begin                      if Length(FCOMENTRecord.CommentString)>=1 then                        begin                          case Ord(FCOMENTRecord.CommentString[1]) of                            CC_OmfExtension_IMPDEF:                              if not ReadImpDef(FCOMENTRecord,objdata) then                                exit;                            CC_OmfExtension_EXPDEF:                              if not ReadExpDef(FCOMENTRecord,objdata) then                                exit;                          end;                        end;                    end;                  CC_LIBMOD:                    begin                      {todo: do we need to read the module name here?}                    end;                  CC_EXESTR:                    begin                      InputError('EXESTR record (Executable String Record) is not supported');                      exit;                    end;                  CC_INCERR:                    begin                      InputError('Invalid object file (contains indication of error encountered during incremental compilation)');                      exit;                    end;                  CC_NOPAD:                    begin                      InputError('NOPAD (No Segment Padding) record is not supported');                      exit;                    end;                  CC_WKEXT:                    begin                      InputError('Weak externals are not supported');                      exit;                    end;                  CC_LZEXT:                    begin                      InputError('Lazy externals are not supported');                      exit;                    end;                  else                    begin                      {the rest are ignored for now...}                    end;                end;              end;            RT_EXTDEF:              if not ReadExtDef(FRawRecord,objdata) then                exit;            RT_LPUBDEF,RT_LPUBDEF32,            RT_PUBDEF,RT_PUBDEF32:              if not ReadPubDef(FRawRecord,objdata) then                exit;            RT_LEDATA,RT_LEDATA32,            RT_LIDATA,RT_LIDATA32,            RT_FIXUPP,RT_FIXUPP32:              if not ReadLeOrLiDataAndFixups(FRawRecord,objdata) then                exit;            RT_MODEND,RT_MODEND32:              if not ReadModEnd(FRawRecord,objdata) then                exit;            RT_LINNUM,RT_LINNUM32:              ;            else              begin                InputError('Unsupported OMF record type $'+HexStr(FRawRecord.RecordType,2));                exit;              end;          end;        until FRawRecord.RecordType in [RT_MODEND,RT_MODEND32];        result:=true;      end;{****************************************************************************                               TMZExeHeader****************************************************************************}    procedure TMZExeHeader.SetHeaderSizeAlignment(AValue: Integer);      begin        if (AValue<16) or ((AValue mod 16) <> 0) then          Internalerror(2015060601);        FHeaderSizeAlignment:=AValue;      end;    constructor TMZExeHeader.Create;      begin        FHeaderSizeAlignment:=16;      end;    procedure TMZExeHeader.WriteTo(aWriter: TObjectWriter);      var        NumRelocs: Word;        HeaderSizeInBytes: DWord;        HeaderParagraphs: Word;        RelocTableOffset: Word;        BytesInLastBlock: Word;        BlocksInFile: Word;        HeaderBytes: array [0..$1B] of Byte;        RelocBytes: array [0..3] of Byte;        TotalExeSize: DWord;        i: Integer;      begin        NumRelocs:=Length(Relocations);        RelocTableOffset:=$1C+Length(ExtraHeaderData);        HeaderSizeInBytes:=Align(RelocTableOffset+4*NumRelocs,16);        HeaderParagraphs:=HeaderSizeInBytes div 16;        TotalExeSize:=HeaderSizeInBytes+LoadableImageSize;        BlocksInFile:=(TotalExeSize+511) div 512;        BytesInLastBlock:=TotalExeSize mod 512;        HeaderBytes[$00]:=$4D;  { 'M' }        HeaderBytes[$01]:=$5A;  { 'Z' }        unaligned(PUint16(@HeaderBytes[$02{..$03}])^):=NtoLE(uint16(BytesInLastBlock));        unaligned(PUint16(@HeaderBytes[$04{..$05}])^):=NtoLE(uint16(BlocksInFile));        unaligned(PUint16(@HeaderBytes[$06{..$07}])^):=NtoLE(uint16(NumRelocs));        unaligned(PUint16(@HeaderBytes[$08{..$09}])^):=NtoLE(uint16(HeaderParagraphs));        unaligned(PUint16(@HeaderBytes[$0A{..$0B}])^):=NtoLE(uint16(MinExtraParagraphs));        unaligned(PUint16(@HeaderBytes[$0C{..$0D}])^):=NtoLE(uint16(MaxExtraParagraphs));        unaligned(PUint16(@HeaderBytes[$0E{..$0F}])^):=NtoLE(uint16(InitialSS));        unaligned(PUint16(@HeaderBytes[$10{..$11}])^):=NtoLE(uint16(InitialSP));        unaligned(PUint16(@HeaderBytes[$12{..$13}])^):=NtoLE(uint16(Checksum));        unaligned(PUint16(@HeaderBytes[$14{..$15}])^):=NtoLE(uint16(InitialIP));        unaligned(PUint16(@HeaderBytes[$16{..$17}])^):=NtoLE(uint16(InitialCS));        unaligned(PUint16(@HeaderBytes[$18{..$19}])^):=NtoLE(uint16(RelocTableOffset));        unaligned(PUint16(@HeaderBytes[$1A{..$1B}])^):=NtoLE(uint16(OverlayNumber));        aWriter.write(HeaderBytes[0],$1C);        aWriter.write(ExtraHeaderData[0],Length(ExtraHeaderData));        for i:=0 to NumRelocs-1 do          with Relocations[i] do            begin              unaligned(PUint16(@RelocBytes[0{..1}])^):=NtoLE(uint16(offset));              unaligned(PUint16(@RelocBytes[2{..3}])^):=NtoLE(uint16(segment));              aWriter.write(RelocBytes[0],4);            end;        { pad with zeros until the end of header (paragraph aligned) }        aWriter.WriteZeros(HeaderSizeInBytes-aWriter.Size);      end;    procedure TMZExeHeader.AddRelocation(aSegment, aOffset: Word);      begin        SetLength(FRelocations,Length(FRelocations)+1);        with FRelocations[High(FRelocations)] do          begin            segment:=aSegment;            offset:=aOffset;          end;      end;{****************************************************************************                               TMZExeSection****************************************************************************}    procedure TMZExeSection.AddObjSection(objsec: TObjSection; ignoreprops: boolean);      begin        { allow mixing initialized and uninitialized data in the same section          => set ignoreprops=true }        inherited AddObjSection(objsec,true);      end;{****************************************************************************                         TMZExeUnifiedLogicalSegment****************************************************************************}    constructor TMZExeUnifiedLogicalSegment.create(HashObjectList: TFPHashObjectList; const s: TSymStr);      var        Separator: SizeInt;      begin        inherited create(HashObjectList,s);        FObjSectionList:=TFPObjectList.Create(false);        { name format is 'SegName||ClassName' }        Separator:=Pos('||',s);        if Separator>0 then          begin            FSegName:=Copy(s,1,Separator-1);            FSegClass:=Copy(s,Separator+2,Length(s)-Separator-1);          end        else          begin            FSegName:=Name;            FSegClass:='';          end;        { wlink recognizes the stack segment by the class name 'STACK' }        { let's be compatible with wlink }        IsStack:=FSegClass='STACK';      end;    destructor TMZExeUnifiedLogicalSegment.destroy;      begin        FObjSectionList.Free;        inherited destroy;      end;    procedure TMZExeUnifiedLogicalSegment.AddObjSection(ObjSec: TOmfObjSection);      begin        ObjSectionList.Add(ObjSec);        ObjSec.MZExeUnifiedLogicalSegment:=self;        { tlink (and ms link?) use the scStack segment combination to recognize          the stack segment.          let's be compatible with tlink as well }        if ObjSec.Combination=scStack then          IsStack:=True;      end;    procedure TMZExeUnifiedLogicalSegment.CalcMemPos;      var        MinMemPos: qword=high(qword);        MaxMemPos: qword=0;        objsec: TOmfObjSection;        i: Integer;      begin        if ObjSectionList.Count=0 then          internalerror(2015082201);        for i:=0 to ObjSectionList.Count-1 do          begin            objsec:=TOmfObjSection(ObjSectionList[i]);            if objsec.MemPos<MinMemPos then              MinMemPos:=objsec.MemPos;            if (objsec.MemPos+objsec.Size)>MaxMemPos then              MaxMemPos:=objsec.MemPos+objsec.Size;          end;        MemPos:=MinMemPos;        Size:=MaxMemPos-MemPos;      end;    function TMZExeUnifiedLogicalSegment.MemPosStr: string;      begin        Result:=HexStr(MemBasePos shr 4,4)+':'+HexStr((MemPos-MemBasePos),4);      end;{****************************************************************************                         TMZExeUnifiedLogicalGroup****************************************************************************}    constructor TMZExeUnifiedLogicalGroup.create(HashObjectList: TFPHashObjectList; const s: TSymStr);      begin        inherited create(HashObjectList,s);        FSegmentList:=TFPHashObjectList.Create(false);      end;    destructor TMZExeUnifiedLogicalGroup.destroy;      begin        FSegmentList.Free;        inherited destroy;      end;    procedure TMZExeUnifiedLogicalGroup.CalcMemPos;      var        MinMemPos: qword=high(qword);        MaxMemPos: qword=0;        UniSeg: TMZExeUnifiedLogicalSegment;        i: Integer;      begin        if SegmentList.Count=0 then          internalerror(2015082202);        for i:=0 to SegmentList.Count-1 do          begin            UniSeg:=TMZExeUnifiedLogicalSegment(SegmentList[i]);            if UniSeg.MemPos<MinMemPos then              MinMemPos:=UniSeg.MemPos;            if (UniSeg.MemPos+UniSeg.Size)>MaxMemPos then              MaxMemPos:=UniSeg.MemPos+UniSeg.Size;          end;        { align *down* on a paragraph boundary }        MemPos:=(MinMemPos shr 4) shl 4;        Size:=MaxMemPos-MemPos;      end;    function TMZExeUnifiedLogicalGroup.MemPosStr: string;      begin        Result:=HexStr(MemPos shr 4,4)+':'+HexStr(MemPos and $f,4);      end;    procedure TMZExeUnifiedLogicalGroup.AddSegment(UniSeg: TMZExeUnifiedLogicalSegment);      begin        SegmentList.Add(UniSeg.Name,UniSeg);        if UniSeg.PrimaryGroup='' then          UniSeg.PrimaryGroup:=Name;      end;{****************************************************************************                               TMZExeOutput****************************************************************************}    function TMZExeOutput.GetMZFlatContentSection: TMZExeSection;      begin        if not assigned(FMZFlatContentSection) then          FMZFlatContentSection:=TMZExeSection(FindExeSection('.MZ_flat_content'));        result:=FMZFlatContentSection;      end;    procedure TMZExeOutput.CalcDwarfUnifiedLogicalSegmentsForSection(const SecName: TSymStr);      var        ExeSec: TMZExeSection;        ObjSec: TOmfObjSection;        UniSeg: TMZExeUnifiedLogicalSegment;        i: Integer;      begin        ExeSec:=TMZExeSection(FindExeSection(SecName));        for i:=0 to ExeSec.ObjSectionList.Count-1 do          begin            ObjSec:=TOmfObjSection(ExeSec.ObjSectionList[i]);            UniSeg:=TMZExeUnifiedLogicalSegment(DwarfUnifiedLogicalSegments.Find(ObjSec.Name));            if not assigned(UniSeg) then              begin                UniSeg:=TMZExeUnifiedLogicalSegment.Create(DwarfUnifiedLogicalSegments,ObjSec.Name);                UniSeg.MemPos:=0;              end;            UniSeg.AddObjSection(ObjSec);          end;        for i:=0 to DwarfUnifiedLogicalSegments.Count-1 do          begin            UniSeg:=TMZExeUnifiedLogicalSegment(DwarfUnifiedLogicalSegments[i]);            UniSeg.CalcMemPos;          end;      end;    procedure TMZExeOutput.CalcExeUnifiedLogicalSegments;      var        ExeSec: TMZExeSection;        ObjSec: TOmfObjSection;        UniSeg: TMZExeUnifiedLogicalSegment;        i: Integer;      begin        ExeSec:=MZFlatContentSection;        for i:=0 to ExeSec.ObjSectionList.Count-1 do          begin            ObjSec:=TOmfObjSection(ExeSec.ObjSectionList[i]);            UniSeg:=TMZExeUnifiedLogicalSegment(ExeUnifiedLogicalSegments.Find(ObjSec.Name));            if not assigned(UniSeg) then              UniSeg:=TMZExeUnifiedLogicalSegment.Create(ExeUnifiedLogicalSegments,ObjSec.Name);            UniSeg.AddObjSection(ObjSec);          end;        for i:=0 to ExeUnifiedLogicalSegments.Count-1 do          begin            UniSeg:=TMZExeUnifiedLogicalSegment(ExeUnifiedLogicalSegments[i]);            UniSeg.CalcMemPos;            if UniSeg.Size>$10000 then              begin                if current_settings.x86memorymodel=mm_tiny then                  Message1(link_e_program_segment_too_large,IntToStr(UniSeg.Size-$10000))                else if UniSeg.SegClass='CODE' then                  Message2(link_e_code_segment_too_large,UniSeg.SegName,IntToStr(UniSeg.Size-$10000))                else if UniSeg.SegClass='DATA' then                  Message2(link_e_data_segment_too_large,UniSeg.SegName,IntToStr(UniSeg.Size-$10000))                else                  Message2(link_e_segment_too_large,UniSeg.SegName,IntToStr(UniSeg.Size-$10000)+' '+UniSeg.SegName);              end;          end;      end;    procedure TMZExeOutput.CalcExeGroups;        procedure AddToGroup(UniSeg:TMZExeUnifiedLogicalSegment;GroupName:TSymStr);          var            Group: TMZExeUnifiedLogicalGroup;          begin            Group:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups.Find(GroupName));            if not assigned(Group) then              Group:=TMZExeUnifiedLogicalGroup.Create(ExeUnifiedLogicalGroups,GroupName);            Group.AddSegment(UniSeg);          end;      var        objdataidx,groupidx,secidx: Integer;        ObjData: TObjData;        ObjGroup: TObjSectionGroup;        ObjSec: TOmfObjSection;        UniGrp: TMZExeUnifiedLogicalGroup;      begin        for objdataidx:=0 to ObjDataList.Count-1 do          begin            ObjData:=TObjData(ObjDataList[objdataidx]);            if assigned(ObjData.GroupsList) then              for groupidx:=0 to ObjData.GroupsList.Count-1 do                begin                  ObjGroup:=TObjSectionGroup(ObjData.GroupsList[groupidx]);                  for secidx:=low(ObjGroup.members) to high(ObjGroup.members) do                    begin                      ObjSec:=TOmfObjSection(ObjGroup.members[secidx]);                      if assigned(ObjSec.MZExeUnifiedLogicalSegment) then                        AddToGroup(ObjSec.MZExeUnifiedLogicalSegment,ObjGroup.Name);                    end;                end;          end;        for groupidx:=0 to ExeUnifiedLogicalGroups.Count-1 do          begin            UniGrp:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups[groupidx]);            UniGrp.CalcMemPos;            if UniGrp.Size>$10000 then              begin                if current_settings.x86memorymodel=mm_tiny then                  Message1(link_e_program_segment_too_large,IntToStr(UniGrp.Size-$10000))                else if UniGrp.Name='DGROUP' then                  Message2(link_e_data_segment_too_large,UniGrp.Name,IntToStr(UniGrp.Size-$10000))                else                  Message2(link_e_group_too_large,UniGrp.Name,IntToStr(UniGrp.Size-$10000));              end;          end;      end;    procedure TMZExeOutput.CalcSegments_MemBasePos;      var        lastbase:qword=0;        i: Integer;        UniSeg: TMZExeUnifiedLogicalSegment;      begin        for i:=0 to ExeUnifiedLogicalSegments.Count-1 do          begin            UniSeg:=TMZExeUnifiedLogicalSegment(ExeUnifiedLogicalSegments[i]);            if (UniSeg.PrimaryGroup<>'') or (UniSeg.IsStack) or               (((UniSeg.MemPos+UniSeg.Size-1)-lastbase)>$ffff) then              lastbase:=(UniSeg.MemPos shr 4) shl 4;            UniSeg.MemBasePos:=lastbase;          end;      end;    procedure TMZExeOutput.WriteMap_SegmentsAndGroups;      var        i, LongestGroupName, LongestSegmentName, LongestClassName: Integer;        UniSeg: TMZExeUnifiedLogicalSegment;        UniGrp: TMZExeUnifiedLogicalGroup;        GroupColumnSize, SegmentColumnSize, ClassColumnSize: LongInt;      begin        LongestGroupName:=0;        for i:=0 to ExeUnifiedLogicalGroups.Count-1 do          begin            UniGrp:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups[i]);            LongestGroupName:=max(LongestGroupName,Length(UniGrp.Name));          end;        LongestSegmentName:=0;        LongestClassName:=0;        for i:=0 to ExeUnifiedLogicalSegments.Count-1 do          begin            UniSeg:=TMZExeUnifiedLogicalSegment(ExeUnifiedLogicalSegments[i]);            LongestSegmentName:=max(LongestSegmentName,Length(UniSeg.SegName));            LongestClassName:=max(LongestClassName,Length(UniSeg.SegClass));          end;        GroupColumnSize:=max(32,LongestGroupName+1);        SegmentColumnSize:=max(23,LongestSegmentName+1);        ClassColumnSize:=max(15,LongestClassName+1);        exemap.AddHeader('Groups list');        exemap.Add('');        exemap.Add(PadSpace('Group',GroupColumnSize)+PadSpace('Address',21)+'Size');        exemap.Add(PadSpace('=====',GroupColumnSize)+PadSpace('=======',21)+'====');        exemap.Add('');        for i:=0 to ExeUnifiedLogicalGroups.Count-1 do          begin            UniGrp:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups[i]);            exemap.Add(PadSpace(UniGrp.Name,GroupColumnSize)+PadSpace(UniGrp.MemPosStr,21)+HexStr(UniGrp.Size,8));          end;        exemap.Add('');        GroupColumnSize:=max(15,LongestGroupName+1);        exemap.AddHeader('Segments list');        exemap.Add('');        exemap.Add(PadSpace('Segment',SegmentColumnSize)+PadSpace('Class',ClassColumnSize)+PadSpace('Group',GroupColumnSize)+PadSpace('Address',16)+'Size');        exemap.Add(PadSpace('=======',SegmentColumnSize)+PadSpace('=====',ClassColumnSize)+PadSpace('=====',GroupColumnSize)+PadSpace('=======',16)+'====');        exemap.Add('');        for i:=0 to ExeUnifiedLogicalSegments.Count-1 do          begin            UniSeg:=TMZExeUnifiedLogicalSegment(ExeUnifiedLogicalSegments[i]);            exemap.Add(PadSpace(UniSeg.SegName,SegmentColumnSize)+PadSpace(UniSeg.SegClass,ClassColumnSize)+PadSpace(UniSeg.PrimaryGroup,GroupColumnSize)+PadSpace(UniSeg.MemPosStr,16)+HexStr(UniSeg.Size,8));          end;        exemap.Add('');      end;    procedure TMZExeOutput.WriteMap_HeaderData;      begin        exemap.AddHeader('Header data');        exemap.Add('Loadable image size: '+HexStr(Header.LoadableImageSize,8));        exemap.Add('Min extra paragraphs: '+HexStr(Header.MinExtraParagraphs,4));        exemap.Add('Max extra paragraphs: '+HexStr(Header.MaxExtraParagraphs,4));        exemap.Add('Initial stack pointer: '+HexStr(Header.InitialSS,4)+':'+HexStr(Header.InitialSP,4));        exemap.Add('Entry point address: '+HexStr(Header.InitialCS,4)+':'+HexStr(Header.InitialIP,4));      end;    function TMZExeOutput.FindStackSegment: TMZExeUnifiedLogicalSegment;      var        i: Integer;        stackseg_wannabe: TMZExeUnifiedLogicalSegment;      begin        Result:=nil;        for i:=0 to ExeUnifiedLogicalSegments.Count-1 do          begin            stackseg_wannabe:=TMZExeUnifiedLogicalSegment(ExeUnifiedLogicalSegments[i]);            { if there are multiple stack segments, choose the largest one.              In theory, we're probably supposed to combine them all and put              them in a contiguous location in memory, but we don't care }            if stackseg_wannabe.IsStack and               (not assigned(result) or (Result.Size<stackseg_wannabe.Size)) then              Result:=stackseg_wannabe;          end;      end;    procedure TMZExeOutput.FillLoadableImageSize;      var        i: Integer;        ExeSec: TMZExeSection;        ObjSec: TOmfObjSection;      begin        Header.LoadableImageSize:=0;        ExeSec:=MZFlatContentSection;        for i:=0 to ExeSec.ObjSectionList.Count-1 do          begin            ObjSec:=TOmfObjSection(ExeSec.ObjSectionList[i]);            if (ObjSec.Size>0) and assigned(ObjSec.Data) then              if (ObjSec.MemPos+ObjSec.Size)>Header.LoadableImageSize then                Header.LoadableImageSize:=ObjSec.MemPos+ObjSec.Size;          end;      end;    procedure TMZExeOutput.FillMinExtraParagraphs;      var        ExeSec: TMZExeSection;      begin        ExeSec:=MZFlatContentSection;        Header.MinExtraParagraphs:=(align(ExeSec.Size,16)-align(Header.LoadableImageSize,16)) div 16;      end;    procedure TMZExeOutput.FillMaxExtraParagraphs;      var        heapmin_paragraphs: Integer;        heapmax_paragraphs: Integer;      begin        if current_settings.x86memorymodel in x86_far_data_models then          begin            { calculate the additional number of paragraphs needed }            heapmin_paragraphs:=(heapsize + 15) div 16;            heapmax_paragraphs:=(maxheapsize + 15) div 16;            Header.MaxExtraParagraphs:=min(Header.MinExtraParagraphs-heapmin_paragraphs+heapmax_paragraphs,$FFFF);          end        else          Header.MaxExtraParagraphs:=$FFFF;      end;    procedure TMZExeOutput.FillStartAddress;      var        EntryMemPos: qword;        EntryMemBasePos: qword;      begin        EntryMemPos:=EntrySym.address;        if assigned(EntrySym.group) then          EntryMemBasePos:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups.Find(EntrySym.group.Name)).MemPos        else          EntryMemBasePos:=TOmfObjSection(EntrySym.objsection).MZExeUnifiedLogicalSegment.MemBasePos;        Header.InitialIP:=EntryMemPos-EntryMemBasePos;        Header.InitialCS:=EntryMemBasePos shr 4;      end;    procedure TMZExeOutput.FillStackAddress;      var        stackseg: TMZExeUnifiedLogicalSegment;      begin        stackseg:=FindStackSegment;        if assigned(stackseg) then          begin            Header.InitialSS:=stackseg.MemBasePos shr 4;            Header.InitialSP:=stackseg.MemPos+stackseg.Size-stackseg.MemBasePos;          end        else          begin            Header.InitialSS:=0;            Header.InitialSP:=0;          end;      end;    procedure TMZExeOutput.FillHeaderData;      begin        Header.MaxExtraParagraphs:=$FFFF;        FillLoadableImageSize;        FillMinExtraParagraphs;        FillMaxExtraParagraphs;        FillStartAddress;        FillStackAddress;        if assigned(exemap) then          WriteMap_HeaderData;      end;    function TMZExeOutput.writeExe: boolean;      var        ExeSec: TMZExeSection;        i: Integer;        ObjSec: TOmfObjSection;      begin        Result:=False;        FillHeaderData;        Header.WriteTo(FWriter);        ExeSec:=MZFlatContentSection;        ExeSec.DataPos:=FWriter.Size;        for i:=0 to ExeSec.ObjSectionList.Count-1 do          begin            ObjSec:=TOmfObjSection(ExeSec.ObjSectionList[i]);            if ObjSec.MemPos<Header.LoadableImageSize then              begin                FWriter.WriteZeros(max(0,ObjSec.MemPos-FWriter.Size+ExeSec.DataPos));                if assigned(ObjSec.Data) then                  FWriter.writearray(ObjSec.Data);              end;          end;        Result:=True;      end;    function TMZExeOutput.writeCom: boolean;      const        ComFileOffset=$100;      var        i: Integer;        ExeSec: TMZExeSection;        ObjSec: TOmfObjSection;        buf: array [0..1023] of byte;        bytesread: LongWord;      begin        FillHeaderData;        if Length(Header.Relocations)>0 then          begin            Message(link_e_com_program_uses_segment_relocations);            exit(False);          end;        ExeSec:=MZFlatContentSection;        for i:=0 to ExeSec.ObjSectionList.Count-1 do          begin            ObjSec:=TOmfObjSection(ExeSec.ObjSectionList[i]);            if ObjSec.MemPos<Header.LoadableImageSize then              begin                FWriter.WriteZeros(max(0,int64(ObjSec.MemPos)-ComFileOffset-int64(FWriter.Size)));                if assigned(ObjSec.Data) then                  begin                    if ObjSec.MemPos<ComFileOffset then                      begin                        ObjSec.Data.seek(ComFileOffset-ObjSec.MemPos);                        repeat                          bytesread:=ObjSec.Data.read(buf,sizeof(buf));                          if bytesread<>0 then                            FWriter.write(buf,bytesread);                        until bytesread=0;                      end                    else                      FWriter.writearray(ObjSec.Data);                  end;              end;          end;        Result:=True;      end;    function TMZExeOutput.writeDebugElf: boolean;      label        cleanup;      var        debugsections: array of TMZExeSection;        debugsections_count: Word;        elfsections_count: Word;        elfsechdrs: array of TElf32sechdr;        shstrndx: Word;        next_section_ofs, elf_start_pos, elf_end_pos: LongWord;        ElfHeader: TElf32header;        shstrtabsect_data: TDynamicArray=Nil;        I, elfsecidx, J: Integer;        ObjSec: TOmfObjSection;        tis_trailer: TTISTrailer;      begin        debugsections:=nil;        elfsechdrs:=nil;                { mark the offset of the start of the ELF image }        elf_start_pos:=Writer.Size;        { count the debug sections }        debugsections_count:=0;        for I:=0 to ExeSectionList.Count-1 do          if oso_debug in TMZExeSection(ExeSectionList[I]).SecOptions then            Inc(debugsections_count);        { extract them into the debugsections array }        SetLength(debugsections,debugsections_count);        debugsections_count:=0;        for I:=0 to ExeSectionList.Count-1 do          if oso_debug in TMZExeSection(ExeSectionList[I]).SecOptions then            begin              debugsections[debugsections_count]:=TMZExeSection(ExeSectionList[I]);              Inc(debugsections_count);            end;        { prepare/allocate elf section headers }        elfsections_count:=debugsections_count+2;        SetLength(elfsechdrs,elfsections_count);        for I:=0 to elfsections_count-1 do          FillChar(elfsechdrs[I],SizeOf(elfsechdrs[I]),0);        shstrndx:=elfsections_count-1;        shstrtabsect_data:=tdynamicarray.Create(SectionDataMaxGrow);        shstrtabsect_data.writestr(#0);        next_section_ofs:=SizeOf(ElfHeader)+elfsections_count*SizeOf(TElf32sechdr);        for I:=0 to debugsections_count-1 do          begin            elfsecidx:=I+1;            with elfsechdrs[elfsecidx] do              begin                sh_name:=shstrtabsect_data.Pos;                sh_type:=SHT_PROGBITS;                sh_flags:=0;                sh_addr:=0;                sh_offset:=next_section_ofs;                sh_size:=debugsections[I].Size;                sh_link:=0;                sh_info:=0;                sh_addralign:=0;                sh_entsize:=0;              end;            Inc(next_section_ofs,debugsections[I].Size);            shstrtabsect_data.writestr(debugsections[I].Name+#0);          end;        with elfsechdrs[shstrndx] do          begin            sh_name:=shstrtabsect_data.Pos;            shstrtabsect_data.writestr('.shstrtab'#0);            sh_type:=SHT_STRTAB;            sh_flags:=0;            sh_addr:=0;            sh_offset:=next_section_ofs;            sh_size:=shstrtabsect_data.Size;            sh_link:=0;            sh_info:=0;            sh_addralign:=0;            sh_entsize:=0;          end;        { write header }        FillChar(ElfHeader,SizeOf(ElfHeader),0);        ElfHeader.e_ident[EI_MAG0]:=ELFMAG0; { = #127'ELF' }        ElfHeader.e_ident[EI_MAG1]:=ELFMAG1;        ElfHeader.e_ident[EI_MAG2]:=ELFMAG2;        ElfHeader.e_ident[EI_MAG3]:=ELFMAG3;        ElfHeader.e_ident[EI_CLASS]:=ELFCLASS32;        ElfHeader.e_ident[EI_DATA]:=ELFDATA2LSB;        ElfHeader.e_ident[EI_VERSION]:=1;        ElfHeader.e_ident[EI_OSABI]:=ELFOSABI_NONE;        ElfHeader.e_ident[EI_ABIVERSION]:=0;        ElfHeader.e_type:=ET_EXEC;        ElfHeader.e_machine:=EM_386;        ElfHeader.e_version:=1;        ElfHeader.e_entry:=0;        ElfHeader.e_phoff:=0;        ElfHeader.e_shoff:=SizeOf(ElfHeader);        ElfHeader.e_flags:=0;        ElfHeader.e_ehsize:=SizeOf(ElfHeader);        ElfHeader.e_phentsize:=SizeOf(TElf32proghdr);        ElfHeader.e_phnum:=0;        ElfHeader.e_shentsize:=SizeOf(TElf32sechdr);        ElfHeader.e_shnum:=elfsections_count;        ElfHeader.e_shstrndx:=shstrndx;        MaybeSwapHeader(ElfHeader);        Writer.write(ElfHeader,sizeof(ElfHeader));        { write section headers }        for I:=0 to elfsections_count-1 do          begin            MaybeSwapSecHeader(elfsechdrs[I]);            Writer.write(elfsechdrs[I],SizeOf(elfsechdrs[I]));          end;        { write section data }        for J:=0 to debugsections_count-1 do          begin            debugsections[J].DataPos:=Writer.Size;            for i:=0 to debugsections[J].ObjSectionList.Count-1 do              begin                ObjSec:=TOmfObjSection(debugsections[J].ObjSectionList[i]);                if assigned(ObjSec.Data) then                  FWriter.writearray(ObjSec.Data);              end;          end;        { write .shstrtab section data }        Writer.writearray(shstrtabsect_data);        { mark the offset past the end of the ELF image }        elf_end_pos:=Writer.Size;        { write TIS trailer (not part of the ELF image) }        FillChar(tis_trailer,sizeof(tis_trailer),0);        with tis_trailer do          begin            tis_signature:=TIS_TRAILER_SIGNATURE;            tis_vendor:=TIS_TRAILER_VENDOR_TIS;            tis_type:=TIS_TRAILER_TYPE_TIS_DWARF;            tis_size:=(elf_end_pos-elf_start_pos)+sizeof(tis_trailer);          end;        MayBeSwapTISTrailer(tis_trailer);        Writer.write(tis_trailer,sizeof(tis_trailer));        Result:=True;cleanup:        shstrtabsect_data.Free;      end;    procedure TMZExeOutput.Load_Symbol(const aname: string);      var        dgroup: TObjSectionGroup;        sym: TObjSymbol;      begin        { special handling for the '_edata' and '_end' symbols, which are          internally added by the linker }        if (aname='_edata') or (aname='_end') then          begin            { create an internal segment with the 'BSS' class }            internalObjData.createsection('*'+aname+'||BSS',0,[]);            { add to group 'DGROUP' }            dgroup:=nil;            if assigned(internalObjData.GroupsList) then              dgroup:=TObjSectionGroup(internalObjData.GroupsList.Find('DGROUP'));            if dgroup=nil then              dgroup:=internalObjData.createsectiongroup('DGROUP');            SetLength(dgroup.members,Length(dgroup.members)+1);            dgroup.members[Length(dgroup.members)-1]:=internalObjData.CurrObjSec;            { define the symbol itself }            sym:=internalObjData.SymbolDefine(aname,AB_GLOBAL,AT_DATA);            sym.group:=dgroup;          end        else          inherited;      end;    procedure TMZExeOutput.DoRelocationFixup(objsec: TObjSection);      var        i: Integer;        omfsec: TOmfObjSection absolute objsec;        objreloc: TOmfRelocation;        target: DWord;        framebase: DWord;        fixupamount: Integer;        target_group: TMZExeUnifiedLogicalGroup;        procedure FixupOffset;          var            w: Word;          begin            omfsec.Data.seek(objreloc.DataOffset);            omfsec.Data.read(w,2);            w:=LEtoN(w);            Inc(w,fixupamount);            w:=NtoLE(w);            omfsec.Data.seek(objreloc.DataOffset);            omfsec.Data.write(w,2);          end;        procedure FixupOffset32;          var            lw: LongWord;          begin            omfsec.Data.seek(objreloc.DataOffset);            omfsec.Data.read(lw,4);            lw:=LEtoN(lw);            Inc(lw,fixupamount);            lw:=NtoLE(lw);            omfsec.Data.seek(objreloc.DataOffset);            omfsec.Data.write(lw,4);          end;        procedure FixupBase(DataOffset: LongWord);          var            w: Word;          begin            omfsec.Data.seek(DataOffset);            omfsec.Data.read(w,2);            w:=LEtoN(w);            Inc(w,framebase shr 4);            w:=NtoLE(w);            omfsec.Data.seek(DataOffset);            omfsec.Data.write(w,2);            Header.AddRelocation(omfsec.MZExeUnifiedLogicalSegment.MemBasePos shr 4,              omfsec.MemPos+DataOffset-omfsec.MZExeUnifiedLogicalSegment.MemBasePos);          end;      begin        for i:=0 to objsec.ObjRelocations.Count-1 do          begin            objreloc:=TOmfRelocation(objsec.ObjRelocations[i]);            if assigned(objreloc.symbol) then              begin                target:=objreloc.symbol.address;                if objreloc.FrameGroup<>'' then                  framebase:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups.Find(objreloc.FrameGroup)).MemPos                else if assigned(objreloc.symbol.group) then                  framebase:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups.Find(objreloc.symbol.group.Name)).MemPos                else                  if assigned(TOmfObjSection(objreloc.symbol.objsection).MZExeUnifiedLogicalSegment) then                    framebase:=TOmfObjSection(objreloc.symbol.objsection).MZExeUnifiedLogicalSegment.MemBasePos                  else                    begin                      framebase:=0;                      Comment(V_Warning,'Encountered an OMF reference to a symbol, that is not present in the final executable: '+objreloc.symbol.Name);                    end;                case objreloc.typ of                  RELOC_ABSOLUTE16,RELOC_ABSOLUTE32,RELOC_SEG,RELOC_FARPTR,RELOC_FARPTR48:                    fixupamount:=target-framebase;                  RELOC_RELATIVE16,RELOC_SEGREL,RELOC_FARPTR_RELATIVEOFFSET:                    fixupamount:=target-(omfsec.MemPos+objreloc.DataOffset)-2;                  RELOC_RELATIVE32,RELOC_FARPTR48_RELATIVEOFFSET:                    fixupamount:=target-(omfsec.MemPos+objreloc.DataOffset)-4;                  else                    internalerror(2015082402);                end;                case objreloc.typ of                  RELOC_ABSOLUTE16,                  RELOC_RELATIVE16:                    FixupOffset;                  RELOC_ABSOLUTE32,                  RELOC_RELATIVE32:                    FixupOffset32;                  RELOC_SEG,                  RELOC_SEGREL:                    FixupBase(objreloc.DataOffset);                  RELOC_FARPTR,                  RELOC_FARPTR_RELATIVEOFFSET:                    begin                      FixupOffset;                      FixupBase(objreloc.DataOffset+2);                    end;                  RELOC_FARPTR48,                  RELOC_FARPTR48_RELATIVEOFFSET:                    begin                      FixupOffset32;                      FixupBase(objreloc.DataOffset+4);                    end;                  else                    internalerror(2015082403);                end;              end            else if assigned(objreloc.objsection) then              begin                target:=objreloc.objsection.MemPos;                if objreloc.FrameGroup<>'' then                  framebase:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups.Find(objreloc.FrameGroup)).MemPos                else                  begin                    if assigned(TOmfObjSection(objreloc.objsection).MZExeUnifiedLogicalSegment) then                      framebase:=TOmfObjSection(objreloc.objsection).MZExeUnifiedLogicalSegment.MemBasePos                    else                      begin                        framebase:=0;                        Comment(V_Warning,'Encountered an OMF reference to a section, that is not present in the final executable: '+TOmfObjSection(objreloc.objsection).Name);                      end;                  end;                case objreloc.typ of                  RELOC_ABSOLUTE16,RELOC_ABSOLUTE32,RELOC_SEG,RELOC_FARPTR,RELOC_FARPTR48:                    fixupamount:=target-framebase;                  RELOC_RELATIVE16,RELOC_SEGREL,RELOC_FARPTR_RELATIVEOFFSET:                    fixupamount:=target-(omfsec.MemPos+objreloc.DataOffset)-2;                  RELOC_RELATIVE32,RELOC_FARPTR48_RELATIVEOFFSET:                    fixupamount:=target-(omfsec.MemPos+objreloc.DataOffset)-4;                  else                    internalerror(2015082405);                end;                case objreloc.typ of                  RELOC_ABSOLUTE16,                  RELOC_RELATIVE16:                    FixupOffset;                  RELOC_ABSOLUTE32,                  RELOC_RELATIVE32:                    FixupOffset32;                  RELOC_SEG,                  RELOC_SEGREL:                    FixupBase(objreloc.DataOffset);                  RELOC_FARPTR,                  RELOC_FARPTR_RELATIVEOFFSET:                    begin                      FixupOffset;                      FixupBase(objreloc.DataOffset+2);                    end;                  RELOC_FARPTR48,                  RELOC_FARPTR48_RELATIVEOFFSET:                    begin                      FixupOffset32;                      FixupBase(objreloc.DataOffset+4);                    end;                  else                    internalerror(2015082406);                end;              end            else if assigned(objreloc.group) then              begin                target_group:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups.Find(objreloc.group.Name));                target:=target_group.MemPos;                if objreloc.FrameGroup<>'' then                  framebase:=TMZExeUnifiedLogicalGroup(ExeUnifiedLogicalGroups.Find(objreloc.FrameGroup)).MemPos                else                  framebase:=target_group.MemPos;                case objreloc.typ of                  RELOC_ABSOLUTE16,RELOC_ABSOLUTE32,RELOC_SEG,RELOC_FARPTR,RELOC_FARPTR48:                    fixupamount:=target-framebase;                  RELOC_RELATIVE16,RELOC_SEGREL,RELOC_FARPTR_RELATIVEOFFSET:                    fixupamount:=target-(omfsec.MemPos+objreloc.DataOffset)-2;                  RELOC_RELATIVE32,RELOC_FARPTR48_RELATIVEOFFSET:                    fixupamount:=target-(omfsec.MemPos+objreloc.DataOffset)-4;                  else                    internalerror(2015111202);                end;                case objreloc.typ of                  RELOC_ABSOLUTE16,                  RELOC_RELATIVE16:                    FixupOffset;                  RELOC_ABSOLUTE32,                  RELOC_RELATIVE32:                    FixupOffset32;                  RELOC_SEG,                  RELOC_SEGREL:                    FixupBase(objreloc.DataOffset);                  RELOC_FARPTR,                  RELOC_FARPTR_RELATIVEOFFSET:                    begin                      FixupOffset;                      FixupBase(objreloc.DataOffset+2);                    end;                  RELOC_FARPTR48,                  RELOC_FARPTR48_RELATIVEOFFSET:                    begin                      FixupOffset32;                      FixupBase(objreloc.DataOffset+4);                    end;                  else                    internalerror(2015111203);                end;              end            else              internalerror(2015082407);          end;      end;    function IOmfObjSectionClassNameCompare(Item1, Item2: Pointer): Integer;      var        I1 : TOmfObjSection absolute Item1;        I2 : TOmfObjSection absolute Item2;      begin        Result:=CompareStr(I1.ClassName,I2.ClassName);        if Result=0 then          Result:=CompareStr(I1.Name,I2.Name);        if Result=0 then          Result:=I1.SortOrder-I2.SortOrder;      end;    procedure TMZExeOutput.Order_ObjSectionList(ObjSectionList: TFPObjectList; const aPattern: string);      var        i: Integer;      begin        for i:=0 to ObjSectionList.Count-1 do          TOmfObjSection(ObjSectionList[i]).SortOrder:=i;        ObjSectionList.Sort(@IOmfObjSectionClassNameCompare);      end;    procedure TMZExeOutput.MemPos_ExeSection(const aname: string);      begin        { overlay all .exe sections on top of each other. In practice, the MZ          formats doesn't have sections, so really, everything goes to a single          section, called .MZ_flat_content. All the remaining sections, that we          use are the debug sections, which go to a separate ELF file, appended          after the end of the .exe. They live in a separate address space, with          each section starting at virtual offset 0. So, that's why we always          set CurrMemPos to 0 before each section here. }        CurrMemPos:=0;        inherited MemPos_ExeSection(aname);      end;    procedure TMZExeOutput.MemPos_EndExeSection;      var        SecName: TSymStr='';      begin        if assigned(CurrExeSec) then          SecName:=CurrExeSec.Name;        inherited MemPos_EndExeSection;        case SecName of          '.MZ_flat_content':            begin              CalcExeUnifiedLogicalSegments;              CalcExeGroups;              CalcSegments_MemBasePos;              if assigned(exemap) then                WriteMap_SegmentsAndGroups;            end;          '.debug_info',          '.debug_abbrev',          '.debug_line',          '.debug_aranges':            begin              CalcDwarfUnifiedLogicalSegmentsForSection(SecName);              with TMZExeSection(FindExeSection(SecName)) do                SecOptions:=SecOptions+[oso_debug];            end;          '':            {nothing to do};          else            internalerror(2018061401);        end;      end;    function TMZExeOutput.writeData: boolean;      begin        Result:=False;        if ExeWriteMode in [ewm_exefull,ewm_exeonly] then          begin            if apptype=app_com then              Result:=WriteCom            else              Result:=WriteExe;            if not Result then              exit;          end;        if ((cs_debuginfo in current_settings.moduleswitches) and            (target_dbg.id in [dbg_dwarf2,dbg_dwarf3,dbg_dwarf4])) and           ((ExeWriteMode=ewm_dbgonly) or            ((ExeWriteMode=ewm_exefull) and              not(cs_link_strip in current_settings.globalswitches))) then          Result:=writeDebugElf;      end;    constructor TMZExeOutput.create;      begin        inherited create;        CExeSection:=TMZExeSection;        CObjData:=TOmfObjData;        CObjSymbol:=TOmfObjSymbol;        { "640K ought to be enough for anybody" :) }        MaxMemPos:=$9FFFF;        FExeUnifiedLogicalSegments:=TFPHashObjectList.Create;        FExeUnifiedLogicalGroups:=TFPHashObjectList.Create;        FDwarfUnifiedLogicalSegments:=TFPHashObjectList.Create;        FHeader:=TMZExeHeader.Create;      end;    destructor TMZExeOutput.destroy;      begin        FHeader.Free;        FDwarfUnifiedLogicalSegments.Free;        FExeUnifiedLogicalGroups.Free;        FExeUnifiedLogicalSegments.Free;        inherited destroy;      end;{****************************************************************************                               TNewExeHeader****************************************************************************}    constructor TNewExeHeader.Create;      begin        SetLength(FMsDosStub,High(win16stub)-Low(win16stub)+1);        Move(win16stub[Low(win16stub)],FMsDosStub[0],High(win16stub)-Low(win16stub)+1);        { BP7 identifies itself as linker version 6.1 in the Win16 .exe files it produces }        LinkerVersion:=6;        LinkerRevision:=1;        LogicalSectorAlignmentShiftCount:=8;  { 256-byte logical sectors }        TargetOS:=netoWindows;        ExpectedWindowsVersion:=$0300;        Flags:=[nehfNotWindowAPICompatible,nehfWindowAPICompatible,nehfMultipleData,nehfProtectedModeOnly];        AdditionalFlags:=[];        GangLoadAreaStart:=0;        GangLoadAreaLength:=0;        Reserved:=0;        Reserved2:=0;      end;    procedure TNewExeHeader.WriteTo(aWriter: TObjectWriter);      var        HeaderBytes: array [0..$3F] of Byte;      begin        aWriter.write(MsDosStub[0],Length(MsDosStub));        HeaderBytes[$00]:=$4E;  { 'N' }        HeaderBytes[$01]:=$45;  { 'E' }        HeaderBytes[$02]:=Byte(LinkerVersion);        HeaderBytes[$03]:=Byte(LinkerRevision);        unaligned(PUint16(@HeaderBytes[$04{..$05}])^):=NtoLE(uint16(EntryTableOffset));        unaligned(PUint16(@HeaderBytes[$06{..$07}])^):=NtoLE(uint16(EntryTableLength));        unaligned(PUint32(@HeaderBytes[$08{..$0B}])^):=NtoLE(uint32(Reserved));        unaligned(PUint16(@HeaderBytes[$0C{..$0D}])^):=NtoLE(uint16(Flags));        unaligned(PUint16(@HeaderBytes[$0E{..$0F}])^):=NtoLE(uint16(AutoDataSegmentNumber));        unaligned(PUint16(@HeaderBytes[$10{..$11}])^):=NtoLE(uint16(InitialLocalHeapSize));        unaligned(PUint16(@HeaderBytes[$12{..$13}])^):=NtoLE(uint16(InitialStackSize));        unaligned(PUint16(@HeaderBytes[$14{..$15}])^):=NtoLE(uint16(InitialIP));        unaligned(PUint16(@HeaderBytes[$16{..$17}])^):=NtoLE(uint16(InitialCS));        unaligned(PUint16(@HeaderBytes[$18{..$19}])^):=NtoLE(uint16(InitialSP));        unaligned(PUint16(@HeaderBytes[$1A{..$1B}])^):=NtoLE(uint16(InitialSS));        unaligned(PUint16(@HeaderBytes[$1C{..$1D}])^):=NtoLE(uint16(SegmentTableEntriesCount));        unaligned(PUint16(@HeaderBytes[$1E{..$1F}])^):=NtoLE(uint16(ModuleReferenceTableEntriesCount));        unaligned(PUint16(@HeaderBytes[$20{..$21}])^):=NtoLE(uint16(NonresidentNameTableLength));        unaligned(PUint16(@HeaderBytes[$22{..$23}])^):=NtoLE(uint16(SegmentTableStart));        unaligned(PUint16(@HeaderBytes[$24{..$25}])^):=NtoLE(uint16(ResourceTableStart));        unaligned(PUint16(@HeaderBytes[$26{..$27}])^):=NtoLE(uint16(ResidentNameTableStart));        unaligned(PUint16(@HeaderBytes[$28{..$29}])^):=NtoLE(uint16(ModuleReferenceTableStart));        unaligned(PUint16(@HeaderBytes[$2A{..$2B}])^):=NtoLE(uint16(ImportedNameTableStart));        unaligned(PUint32(@HeaderBytes[$2C{..$2F}])^):=NtoLE(uint32(NonresidentNameTableStart));        unaligned(PUint16(@HeaderBytes[$30{..$31}])^):=NtoLE(uint16(MovableEntryPointsCount));        unaligned(PUint16(@HeaderBytes[$32{..$33}])^):=NtoLE(uint16(LogicalSectorAlignmentShiftCount));        unaligned(PUint16(@HeaderBytes[$34{..$35}])^):=NtoLE(uint16(ResourceSegmentsCount));        HeaderBytes[$36]:=Byte(Ord(TargetOS));        HeaderBytes[$37]:=Byte(AdditionalFlags);        unaligned(PUint16(@HeaderBytes[$38{..$39}])^):=NtoLE(uint16(GangLoadAreaStart));        unaligned(PUint16(@HeaderBytes[$3A{..$3B}])^):=NtoLE(uint16(GangLoadAreaLength));        unaligned(PUint16(@HeaderBytes[$3C{..$3D}])^):=NtoLE(uint16(Reserved2));        unaligned(PUint16(@HeaderBytes[$3E{..$3F}])^):=NtoLE(uint16(ExpectedWindowsVersion));        aWriter.write(HeaderBytes[0],$40);      end;{****************************************************************************                           TNewExeResourceTable****************************************************************************}    function TNewExeResourceTable.GetSize: QWord;      begin        Result:=5;      end;    constructor TNewExeResourceTable.Create;      begin        ResourceDataAlignmentShiftCount:=8;      end;    procedure TNewExeResourceTable.WriteTo(aWriter: TObjectWriter);        procedure WriteAlignShift;          var            AlignShiftBytes: array [0..1] of Byte;          begin            unaligned(PUint16(@AlignShiftBytes[0{..1}])^):=NtoLE(uint16(ResourceDataAlignmentShiftCount));            aWriter.write(AlignShiftBytes[0],2);          end;        procedure WriteEndTypes;          const            EndTypesBytes: array [0..1] of Byte = (0, 0);          begin            aWriter.write(EndTypesBytes[0],2);          end;        procedure WriteEndNames;          const            EndNames: Byte = 0;          begin            aWriter.write(EndNames,1);          end;      begin        WriteAlignShift;        WriteEndTypes;        WriteEndNames;      end;{****************************************************************************                       TNewExeExportNameTableEntry****************************************************************************}    constructor TNewExeExportNameTableEntry.Create(HashObjectList:TFPHashObjectList;const s:TSymStr;OrdNr:Word);      begin        inherited Create(HashObjectList,s);        OrdinalNr:=OrdNr;      end;{****************************************************************************                         TNewExeExportNameTable****************************************************************************}    function TNewExeExportNameTable.GetSize: QWord;      var        i: Integer;      begin        { the end of table mark is 1 byte }        Result:=1;        { each entry is 3 bytes, plus the length of the name }        for i:=0 to Count-1 do          Inc(Result,3+Length(TNewExeExportNameTableEntry(Items[i]).Name));      end;    procedure TNewExeExportNameTable.WriteTo(aWriter: TObjectWriter);      var        i: Integer;        rn: TNewExeExportNameTableEntry;        slen: Byte;        OrdNrBuf: array [0..1] of Byte;      begin        for i:=0 to Count-1 do          begin            rn:=TNewExeExportNameTableEntry(Items[i]);            slen:=Length(rn.Name);            if slen=0 then              internalerror(2019080801);            aWriter.write(slen,1);            aWriter.write(rn.Name[1],slen);            unaligned(PUint16(@OrdNrBuf[0{..1}])^):=NtoLE(uint16(rn.OrdinalNr));            aWriter.write(OrdNrBuf[0],2);          end;        { end of table mark }        slen:=0;        aWriter.write(slen,1);      end;{****************************************************************************                         TNewExeModuleReferenceTable****************************************************************************}    function TNewExeModuleReferenceTable.GetSize: QWord;      begin        Result:=Count*2;      end;    procedure TNewExeModuleReferenceTable.AddModuleReference(const dllname:TSymStr);      begin        if not Assigned(Find(dllname)) then          TNewExeModuleReferenceTableEntry.Create(Self,dllname);      end;    procedure TNewExeModuleReferenceTable.WriteTo(aWriter: TObjectWriter;imptbl: TNewExeImportedNameTable);      var        buf: array of Byte;        i: Integer;        ImpTblEntry: TNewExeImportedNameTableEntry;      begin        SetLength(buf,Size);        for i:=0 to Count-1 do          begin            ImpTblEntry:=TNewExeImportedNameTableEntry(imptbl.Find(TNewExeModuleReferenceTableEntry(Items[i]).Name));            if not Assigned(ImpTblEntry) then              internalerror(2019080903);            unaligned(PUint16(@buf[2*i{..2*i+1}])^):=NtoLE(uint16(ImpTblEntry.TableOffset));          end;        aWriter.write(buf[0],Length(buf));      end;{****************************************************************************                          TNewExeImportedNameTable****************************************************************************}    function TNewExeImportedNameTable.GetSize: QWord;      var        i: Integer;      begin        { the table starts with an empty entry, which takes 1 byte }        Result:=1;        { each entry is 1 byte, plus the length of the name }        for i:=0 to Count-1 do          Inc(Result,1+Length(TNewExeImportedNameTableEntry(Items[i]).Name));      end;    procedure TNewExeImportedNameTable.AddImportedName(const name: TSymStr);      begin        if not Assigned(Find(name)) then          TNewExeImportedNameTableEntry.Create(Self,name);      end;    procedure TNewExeImportedNameTable.CalcTableOffsets;      var        cofs: LongInt;        i: Integer;        entry: TNewExeImportedNameTableEntry;      begin        { the table starts with an empty entry, which takes 1 byte }        cofs:=1;        for i:=0 to Count-1 do          begin            entry:=TNewExeImportedNameTableEntry(Items[i]);            entry.TableOffset:=cofs;            Inc(cofs,1+Length(entry.Name));            if cofs>High(Word) then              internalerror(2019080902);          end;      end;    procedure TNewExeImportedNameTable.WriteTo(aWriter: TObjectWriter);      var        i: Integer;        entry: TNewExeImportedNameTableEntry;        slen: Byte;      begin        { the table starts with an empty entry }        slen:=0;        aWriter.write(slen,1);        for i:=0 to Count-1 do          begin            entry:=TNewExeImportedNameTableEntry(Items[i]);            slen:=Length(entry.Name);            if slen=0 then              internalerror(2019080901);            aWriter.write(slen,1);            aWriter.write(entry.Name[1],slen);          end;      end;{****************************************************************************                            TNewExeEntryPoint****************************************************************************}    function TNewExeEntryPoint.GetFlagsByte: Byte;      begin        Result:=Byte(ParmCount shl 3);        if neepfExported in Flags then          Result:=Result or 1;        if neepfSingleData in Flags then          Result:=Result or 2;      end;{****************************************************************************                            TNewExeEntryTable****************************************************************************}    function TNewExeEntryTable.GetSize: QWord;      var        CurBundleStart: Integer;        CurBundleSize: Byte;      begin        Result:=0;        CurBundleStart:=1;        repeat          CurBundleSize:=BundleSize(CurBundleStart);          Inc(Result,2);          if CurBundleSize>0 then            begin              if Items[CurBundleStart]=nil then                { a bundle of null entries }              else if neepfMovableSegment in Items[CurBundleStart].Flags then                { a bundle of movable segment records }                Inc(Result,6*CurBundleSize)              else                { a bundle of fixed segment records }                Inc(Result,3*CurBundleSize);            end;          Inc(CurBundleStart,CurBundleSize);        until CurBundleSize=0;      end;    procedure TNewExeEntryTable.SetItems(i: Integer; AValue: TNewExeEntryPoint);      begin        if (i<1) or (i>Length(FItems)) then          internalerror(2019081002);        FItems[i-1]:=AValue;      end;    function TNewExeEntryTable.CanBeInSameBundle(i, j: Integer): Boolean;      begin        if (Items[i]=nil) or (Items[j]=nil) then          Result:=(Items[i]=nil) and (Items[j]=nil)        else if not (neepfMovableSegment in Items[i].Flags) and                not (neepfMovableSegment in Items[j].Flags) then          Result:=Items[i].Segment=Items[j].Segment        else          Result:=(neepfMovableSegment in Items[i].Flags)=                  (neepfMovableSegment in Items[j].Flags);      end;    function TNewExeEntryTable.BundleSize(StartingElement:Integer): Byte;      begin        if StartingElement>Count then          Result:=0        else          begin            Result:=1;            while (Result<255) and ((StartingElement+Result)<=Count) and CanBeInSameBundle(StartingElement,StartingElement+Result) do              Inc(Result);          end;      end;    function TNewExeEntryTable.GetCount: Word;      begin        Result:=Length(FItems);      end;    function TNewExeEntryTable.GetItems(i: Integer): TNewExeEntryPoint;      begin        if (i<1) or (i>Length(FItems)) then          internalerror(2019081011);        Result:=FItems[i-1];      end;    destructor TNewExeEntryTable.Destroy;      var        i: Integer;      begin        for i:=low(FItems) to high(FItems) do          FreeAndNil(FItems[i]);        inherited Destroy;      end;    procedure TNewExeEntryTable.WriteTo(aWriter: TObjectWriter);      var        CurBundleStart, i: Integer;        CurBundleSize: Byte;        buf: array [0..5] of Byte;        cp: TNewExeEntryPoint;      begin        CurBundleStart:=1;        repeat          CurBundleSize:=BundleSize(CurBundleStart);          aWriter.write(CurBundleSize,1);          if CurBundleSize>0 then            begin              if Items[CurBundleStart]=nil then                begin                  { a bundle of null entries }                  buf[0]:=0;                  aWriter.write(buf[0],1);                end              else if neepfMovableSegment in Items[CurBundleStart].Flags then                begin                  { a bundle of movable segment records }                  buf[0]:=$ff;                  aWriter.write(buf[0],1);                  for i:=CurBundleStart to CurBundleStart+CurBundleSize-1 do                    begin                      cp:=Items[i];                      buf[0]:=cp.FlagsByte;                      buf[1]:=$CD;  { INT 3Fh instruction }                      buf[2]:=$3F;                      buf[3]:=Byte(cp.Segment);                      unaligned(PUint16(@buf[4{..5}])^):=NtoLE(uint16(cp.Offset));                      aWriter.write(buf[0],6);                    end;                end              else                begin                  { a bundle of fixed segment records }                  buf[0]:=Items[CurBundleStart].Segment;                  aWriter.write(buf[0],1);                  for i:=CurBundleStart to CurBundleStart+CurBundleSize-1 do                    begin                      cp:=Items[i];                      buf[0]:=cp.FlagsByte;                      unaligned(PUint16(@buf[1{..2}])^):=NtoLE(uint16(cp.Offset));                      aWriter.write(buf[0],3);                    end;                end;            end;          Inc(CurBundleStart,CurBundleSize);        until CurBundleSize=0;        { finish the end marker - a null bundle of 0 entries - must be 2 zero          bytes. The first one was already written by the loop, time to add the          second one. }        buf[0]:=0;        aWriter.write(buf[0],1);      end;    procedure TNewExeEntryTable.GrowTo(aNewCount: Word);      begin        if aNewCount<Count then          internalerror(2019081003);        SetLength(FItems,aNewCount);      end;{****************************************************************************                             TNewExeRelocation****************************************************************************}    procedure TNewExeRelocation.EncodeTo(dest: PByte);      begin        dest[0]:=Ord(AddressType);        dest[1]:=Ord(RelocationType) or (Ord(IsAdditive) shl 2);        unaligned(PUint16(@dest[2{..3}])^):=NtoLE(uint16(Offset));        case RelocationType of          nertInternalRef:            begin              case InternalRefSegmentType of                neirstFixed:                  begin                    dest[4]:=Byte(InternalRefFixedSegmentNumber);                    dest[5]:=0;                    unaligned(PUint16(@dest[6{..7}])^):=NtoLE(uint16(InternalRefFixedSegmentOffset));                  end;                neirstMovable:                  begin                    dest[4]:=$FF;                    dest[5]:=0;                    unaligned(PUint16(@dest[6{..7}])^):=NtoLE(uint16(InternalRefMovableSegmentEntryTableIndex));                  end;              end;            end;          nertImportName:            begin              unaligned(PUint16(@dest[4{..5}])^):=NtoLE(uint16(ImportModuleIndex));              unaligned(PUint16(@dest[6{..7}])^):=NtoLE(uint16(ImportNameIndex));            end;          nertImportOrdinal:            begin              unaligned(PUint16(@dest[4{..5}])^):=NtoLE(uint16(ImportModuleIndex));              unaligned(PUint16(@dest[6{..7}])^):=NtoLE(uint16(ImportOrdinal));            end;          nertOsFixup:            begin              unaligned(PUint16(@dest[4{..5}])^):=NtoLE(uint16(Ord(OsFixupType)));              unaligned(PUint16(@dest[6{..7}])^):=0;            end;        end;      end;{****************************************************************************                           TNewExeRelocationList****************************************************************************}    function TNewExeRelocationList.GetCount: Integer;      begin        Result:=FInternalList.Count;      end;    function TNewExeRelocationList.GetItem(Index: Integer): TNewExeRelocation;      begin        Result:=TNewExeRelocation(FInternalList[Index]);      end;    function TNewExeRelocationList.GetSize: QWord;      begin        Result:=2+Count*NewExeRelocationRecordSize;      end;    procedure TNewExeRelocationList.SetCount(AValue: Integer);      begin        FInternalList.Count:=AValue;      end;    procedure TNewExeRelocationList.SetItem(Index:Integer;AValue:TNewExeRelocation);      begin        FInternalList[Index]:=AValue;      end;    constructor TNewExeRelocationList.Create;      begin        FInternalList:=TFPObjectList.Create;      end;    destructor TNewExeRelocationList.Destroy;      begin        FInternalList.Free;        inherited Destroy;      end;    procedure TNewExeRelocationList.WriteTo(aWriter: TObjectWriter);      var        buf: array of Byte;        p: PByte;        i: Integer;      begin        SetLength(buf,Size);        unaligned(PUint16(@buf[0])^):=NtoLE(uint16(Count));        p:=@(buf[2]);        for i:=0 to Count-1 do          begin            Items[i].EncodeTo(p);            Inc(p,NewExeRelocationRecordSize);          end;        aWriter.write(buf[0],Size);      end;    function TNewExeRelocationList.Add(AObject: TNewExeRelocation): Integer;      begin        Result:=FInternalList.Add(AObject);      end;{****************************************************************************                              TNewExeSection****************************************************************************}    function TNewExeSection.GetMinAllocSize: QWord;      begin        Result:=Size-StackSize;      end;    function TNewExeSection.GetNewExeSegmentFlags: TNewExeSegmentFlags;      begin        Result:=FNewExeSegmentFlags;        if Relocations.Count>0 then          Include(Result,nesfHasRelocationData)        else          Exclude(Result,nesfHasRelocationData);      end;    constructor TNewExeSection.create(AList:TFPHashObjectList;const AName:string);      begin        inherited create(AList, AName);        FRelocations:=TNewExeRelocationList.Create;      end;    destructor TNewExeSection.destroy;      begin        FRelocations.Free;        inherited destroy;      end;    procedure TNewExeSection.WriteHeaderTo(aWriter: TObjectWriter);      var        SegmentHeaderBytes: array [0..7] of Byte;      begin        unaligned(PUint16(@SegmentHeaderBytes[0{..1}])^):=NtoLE(uint16(DataPosSectors));        unaligned(PUint16(@SegmentHeaderBytes[2{..3}])^):=NtoLE(uint16(SizeInFile));        unaligned(PUint16(@SegmentHeaderBytes[4{..5}])^):=NtoLE(uint16(NewExeSegmentFlags));        unaligned(PUint16(@SegmentHeaderBytes[6{..7}])^):=NtoLE(uint16(MinAllocSize));        aWriter.write(SegmentHeaderBytes[0],8);      end;    function TNewExeSection.MemPosStr(AImageBase: qword): string;      begin        Result:=HexStr(MemBasePos,4)+':'+HexStr(MemPos,4);      end;    procedure TNewExeSection.AddObjSection(objsec: TObjSection; ignoreprops: boolean);      var        s: TSymStr;        Separator: SizeInt;        {SegName,} SegClass: string;        IsStack, IsBss: Boolean;      begin        { allow mixing initialized and uninitialized data in the same section          => set ignoreprops=true }        inherited AddObjSection(objsec,true);        IsBss:=not(oso_Data in objsec.SecOptions);        s:=objsec.Name;        { name format is 'SegName||ClassName' }        Separator:=Pos('||',s);        if Separator>0 then          begin            //SegName:=Copy(s,1,Separator-1);            SegClass:=Copy(s,Separator+2,Length(s)-Separator-1);          end        else          begin            //SegName:=s;            SegClass:='';          end;        { wlink recognizes the stack segment by the class name 'STACK' }        { let's be compatible with wlink }        IsStack:=SegClass='STACK';        { tlink (and ms link?) use the scStack segment combination to recognize          the stack segment.          let's be compatible with tlink as well }        if TOmfObjSection(ObjSec).Combination=scStack then          IsStack:=True;        if IsStack then          StackSize:=StackSize+objsec.Size;        EarlySize:=align_qword(EarlySize,SecAlign)+objsec.Size;        if (not IsBss) and (not IsStack) then          SizeInFile:=EarlySize;      end;    function TNewExeSection.CanAddObjSection(objsec: TObjSection; ExeSectionLimit: QWord): boolean;      var        NewSecAlign: LongInt;        NewSize: QWord;      begin        NewSecAlign:=max(objsec.SecAlign,SecAlign);        NewSize:=align_qword(EarlySize,NewSecAlign)+objsec.Size;        Result:=NewSize<=ExeSectionLimit;      end;{****************************************************************************                               TNewExeOutput****************************************************************************}    procedure TNewExeOutput.AddImportSymbol(const libname, symname,      symmangledname: TCmdStr; OrdNr: longint; isvar: boolean);      var        ImportLibrary: TImportLibrary;        ImportSymbol: TFPHashObject;      begin        ImportLibrary:=TImportLibrary(FImports.Find(libname));        if not assigned(ImportLibrary) then          ImportLibrary:=TImportLibrary.Create(FImports,libname);        ImportSymbol:=TFPHashObject(ImportLibrary.ImportSymbolList.Find(symname));        if not assigned(ImportSymbol) then          ImportSymbol:=TImportSymbol.Create(ImportLibrary.ImportSymbolList,symname,symmangledname,OrdNr,isvar);      end;    procedure TNewExeOutput.AddImportLibrariesExtractedFromObjectModules;      var        i, j, k: Integer;        ObjData: TOmfObjData;        ImportLibrary: TImportLibrary;        ImportSymbol: TImportSymbol;      begin        for i:=0 to ObjDataList.Count-1 do          begin            ObjData:=TOmfObjData(ObjDataList[i]);            for j:=0 to ObjData.ImportLibraryList.Count-1 do              begin                ImportLibrary:=TImportLibrary(ObjData.ImportLibraryList[j]);                for k:=0 to ImportLibrary.ImportSymbolList.Count-1 do                  begin                    ImportSymbol:=TImportSymbol(ImportLibrary.ImportSymbolList[k]);                    AddImportSymbol(ImportLibrary.Name,ImportSymbol.Name,ImportSymbol.MangledName,ImportSymbol.OrdNr,ImportSymbol.IsVar);                  end;              end;          end;      end;    procedure TNewExeOutput.AddNewExeSection;      var        SegNr: Integer;        SecName: string;      begin        SegNr:=ExeSectionList.Count+1;        WriteStr(SecName,'Segment',SegNr,'_',NewExeMetaSection2String[CurrExeMetaSec]);        inherited Order_ExeSection(SecName);        TNewExeSection(CurrExeSec).ExeMetaSec:=CurrExeMetaSec;        TNewExeSection(CurrExeSec).MemBasePos:=SegNr;        if (CurrExeMetaSec=nemsData) and (Header.AutoDataSegmentNumber=0) then          Header.AutoDataSegmentNumber:=SegNr;        case CurrExeMetaSec of          nemsCode:            TNewExeSection(CurrExeSec).NewExeSegmentFlags:=[nesfMovable,nesfPreload];          nemsData:            TNewExeSection(CurrExeSec).NewExeSegmentFlags:=[nesfData,nesfPreload];          else            internalerror(2019070601);        end;      end;    function TNewExeOutput.WriteNewExe: boolean;        function ExtractModuleName(filename: string): string;          begin            Result:=UpCase(ChangeFileExt(filename,''));          end;      var        i: Integer;      begin        if IsSharedLibrary then          Header.Flags:=Header.Flags+[nehfIsDLL,nehfSingleData]-[nehfMultipleData];        { all exported symbols must have an ordinal }        AssignOrdinalsToAllExportSymbols;        AddEntryPointsForAllExportSymbols;        { the first entry in the resident-name table is the module name }        TNewExeExportNameTableEntry.Create(ResidentNameTable,ExtractModuleName(current_module.exefilename),0);        { the first entry in the nonresident-name table is the module description }        TNewExeExportNameTableEntry.Create(NonresidentNameTable,description,0);        { add all symbols, exported by name to the resident and nonresident-name tables }        AddExportedNames;        FillImportedNameAndModuleReferenceTable;        ImportedNameTable.CalcTableOffsets;        Header.InitialIP:=EntrySym.address;        Header.InitialCS:=TNewExeSection(EntrySym.objsection.ExeSection).MemBasePos;        Header.InitialSP:=0;        Header.InitialSS:=Header.AutoDataSegmentNumber;        Header.InitialStackSize:=TNewExeSection(ExeSectionList[Header.AutoDataSegmentNumber-1]).StackSize;        Header.InitialLocalHeapSize:=heapsize;        Header.SegmentTableStart:=NewExeHeaderSize;        Header.SegmentTableEntriesCount:=ExeSectionList.Count;        Header.ResourceTableStart:=Header.SegmentTableStart+NewExeSegmentHeaderSize*Header.SegmentTableEntriesCount;        Header.ResidentNameTableStart:=Header.ResourceTableStart+ResourceTable.Size;        Header.ModuleReferenceTableStart:=Header.ResidentNameTableStart+ResidentNameTable.Size;        Header.ModuleReferenceTableEntriesCount:=ModuleReferenceTable.Count;        Header.ImportedNameTableStart:=Header.ModuleReferenceTableStart+ModuleReferenceTable.Size;        Header.EntryTableOffset:=Header.ImportedNameTableStart+ImportedNameTable.Size;        Header.EntryTableLength:=EntryTable.Size;        Header.NonresidentNameTableStart:=Header.EntryTableOffset+Header.EntryTableLength+Length(Header.MsDosStub);        Header.NonresidentNameTableLength:=NonresidentNameTable.Size;        Header.WriteTo(FWriter);        for i:=0 to ExeSectionList.Count-1 do          TNewExeSection(ExeSectionList[i]).WriteHeaderTo(FWriter);        ResourceTable.WriteTo(FWriter);        ResidentNameTable.WriteTo(FWriter);        ModuleReferenceTable.WriteTo(FWriter,ImportedNameTable);        ImportedNameTable.WriteTo(FWriter);        EntryTable.WriteTo(FWriter);        NonresidentNameTable.WriteTo(FWriter);        { todo: write the rest of the file as well }        Result:=True;      end;    procedure TNewExeOutput.FillImportedNameAndModuleReferenceTable;      var        i, j: Integer;        ImportLibrary: TImportLibrary;        ImportSymbol: TImportSymbol;        exesym: TExeSymbol;        LibNameAdded: Boolean;        dllname: TSymStr;      begin        for i:=0 to FImports.Count-1 do          begin            ImportLibrary:=TImportLibrary(FImports[i]);            LibNameAdded:=False;            for j:=0 to ImportLibrary.ImportSymbolList.Count-1 do              begin                ImportSymbol:=TImportSymbol(ImportLibrary.ImportSymbolList[j]);                exesym:=TExeSymbol(ExeSymbolList.Find(ImportSymbol.MangledName));                if assigned(exesym) then                  begin                    if not LibNameAdded then                      begin                        dllname:=StripDllExt(ImportLibrary.Name);                        ImportedNameTable.AddImportedName(dllname);                        ModuleReferenceTable.AddModuleReference(dllname);                        LibNameAdded:=True;                      end;                    if (ImportSymbol.OrdNr=0) and (ImportSymbol.Name<>'') then                      ImportedNameTable.AddImportedName(ImportSymbol.Name);                  end;              end;          end;      end;    function TNewExeOutput.GetHighestExportSymbolOrdinal: Word;      var        i, j: Integer;        ObjData: TOmfObjData;        sym: TOmfObjExportedSymbol;      begin        Result:=0;        for i:=0 to ObjDataList.Count-1 do          begin            ObjData:=TOmfObjData(ObjDataList[i]);            for j:=0 to ObjData.ExportedSymbolList.Count-1 do              begin                sym:=TOmfObjExportedSymbol(ObjData.ExportedSymbolList[j]);                if sym.ExportByOrdinal then                  Result:=Max(Result,sym.ExportOrdinal);              end;          end;      end;    procedure TNewExeOutput.AssignOrdinalsToAllExportSymbols;      var        NextOrdinal: LongInt;        i, j: Integer;        ObjData: TOmfObjData;        sym: TOmfObjExportedSymbol;      begin        NextOrdinal:=GetHighestExportSymbolOrdinal+1;        for i:=0 to ObjDataList.Count-1 do          begin            ObjData:=TOmfObjData(ObjDataList[i]);            for j:=0 to ObjData.ExportedSymbolList.Count-1 do              begin                sym:=TOmfObjExportedSymbol(ObjData.ExportedSymbolList[j]);                if not sym.ExportByOrdinal then                  begin                    if NextOrdinal>High(Word) then                      internalerror(2019081001);                    sym.ExportByOrdinal:=True;                    sym.ExportOrdinal:=NextOrdinal;                    Inc(NextOrdinal);                  end;              end;          end;      end;    procedure TNewExeOutput.AddEntryPointsForAllExportSymbols;      var        LastOrdinal: Word;        i, j: Integer;        ObjData: TOmfObjData;        sym: TOmfObjExportedSymbol;        ent: TNewExeEntryPoint;        exesym: TExeSymbol;        sec: TNewExeSection;      begin        LastOrdinal:=GetHighestExportSymbolOrdinal;        EntryTable.GrowTo(LastOrdinal);        for i:=0 to ObjDataList.Count-1 do          begin            ObjData:=TOmfObjData(ObjDataList[i]);            for j:=0 to ObjData.ExportedSymbolList.Count-1 do              begin                sym:=TOmfObjExportedSymbol(ObjData.ExportedSymbolList[j]);                { all exports must have an ordinal at this point }                if not sym.ExportByOrdinal then                  internalerror(2019081004);                { check for duplicated ordinals }                if Assigned(EntryTable[sym.ExportOrdinal]) then                  internalerror(2019081005);                ent:=TNewExeEntryPoint.Create;                EntryTable[sym.ExportOrdinal]:=ent;                exesym:=TExeSymbol(ExeSymbolList.Find(sym.InternalName));                if not Assigned(exesym) then                  internalerror(2019081006);                ent.Flags:=[neepfExported];                if IsSharedLibrary then                  ent.Flags:=ent.Flags+[neepfSingleData];                ent.Offset:=exesym.ObjSymbol.address;                sec:=TNewExeSection(exesym.ObjSymbol.objsection.ExeSection);                ent.Segment:=sec.MemBasePos;                if nesfMovable in sec.NewExeSegmentFlags then                  ent.Flags:=ent.Flags+[neepfMovableSegment];                ent.ParmCount:=sym.ParmCount;              end;          end;      end;    procedure TNewExeOutput.AddExportedNames;      var        i, j: Integer;        ObjData: TOmfObjData;        sym: TOmfObjExportedSymbol;      begin        for i:=0 to ObjDataList.Count-1 do          begin            ObjData:=TOmfObjData(ObjDataList[i]);            for j:=0 to ObjData.ExportedSymbolList.Count-1 do              begin                sym:=TOmfObjExportedSymbol(ObjData.ExportedSymbolList[j]);                { all exports must have an ordinal at this point }                if not sym.ExportByOrdinal then                  internalerror(2019081007);                if sym.ResidentName then                  TNewExeExportNameTableEntry.Create(ResidentNameTable,sym.ExportedName,sym.ExportOrdinal)                else                  TNewExeExportNameTableEntry.Create(NonresidentNameTable,sym.ExportedName,sym.ExportOrdinal);              end;          end;      end;    procedure TNewExeOutput.DoRelocationFixup(objsec: TObjSection);      begin        {todo}      end;    function INewExeOmfObjSectionClassNameCompare(Item1, Item2: Pointer): Integer;      var        I1 : TOmfObjSection absolute Item1;        I2 : TOmfObjSection absolute Item2;      begin        Result:=CompareStr(I1.ClassName,I2.ClassName);        if Result=0 then          Result:=CompareStr(I1.Name,I2.Name);        if Result=0 then          Result:=I1.SortOrder-I2.SortOrder;      end;    procedure TNewExeOutput.Order_ObjSectionList(ObjSectionList: TFPObjectList;const aPattern: string);      var        i: Integer;      begin        for i:=0 to ObjSectionList.Count-1 do          TOmfObjSection(ObjSectionList[i]).SortOrder:=i;        ObjSectionList.Sort(@INewExeOmfObjSectionClassNameCompare);      end;    constructor TNewExeOutput.create;      begin        inherited create;        CObjData:=TOmfObjData;        CObjSymbol:=TOmfObjSymbol;        CExeSection:=TNewExeSection;        FHeader:=TNewExeHeader.Create;        MaxMemPos:=$FFFFFFFF;        CurrExeMetaSec:=nemsNone;        FResourceTable:=TNewExeResourceTable.Create;        FResidentNameTable:=TNewExeExportNameTable.Create;        FNonresidentNameTable:=TNewExeExportNameTable.Create;        FModuleReferenceTable:=TNewExeModuleReferenceTable.Create;        FImportedNameTable:=TNewExeImportedNameTable.Create;        FEntryTable:=TNewExeEntryTable.Create;      end;    destructor TNewExeOutput.destroy;      begin        FEntryTable.Free;        FImportedNameTable.Free;        FModuleReferenceTable.Free;        FNonresidentNameTable.Free;        FResidentNameTable.Free;        FResourceTable.Free;        FHeader.Free;        inherited destroy;      end;    procedure TNewExeOutput.Order_ExeSection(const aname: string);      begin        case aname of          '.NE_code':            CurrExeMetaSec:=nemsCode;          '.NE_data':            CurrExeMetaSec:=nemsData;          else            internalerror(2019080201);        end;      end;    procedure TNewExeOutput.Order_EndExeSection;      begin        CurrExeMetaSec:=nemsNone;        inherited;      end;    procedure TNewExeOutput.Order_ObjSection(const aname: string);      const        SegmentLimit=$10000;      var        i,j     : longint;        ObjData : TObjData;        objsec  : TObjSection;        TmpObjSectionList : TFPObjectList;      begin        if CurrExeMetaSec=nemsNone then          internalerror(2019080202);        if not assigned (CurrExeSec) then          AddNewExeSection;        TmpObjSectionList:=TFPObjectList.Create(false);        for i:=0 to ObjDataList.Count-1 do          begin            ObjData:=TObjData(ObjDataList[i]);            for j:=0 to ObjData.ObjSectionList.Count-1 do              begin                objsec:=TObjSection(ObjData.ObjSectionList[j]);                if (not objsec.Used) and                   MatchPattern(aname,objsec.name) then                  TmpObjSectionList.Add(objsec);              end;          end;        { Order list if needed }        Order_ObjSectionList(TmpObjSectionList,aname);        { Add the (ordered) list to the current ExeSection }        for i:=0 to TmpObjSectionList.Count-1 do          begin            objsec:=TObjSection(TmpObjSectionList[i]);            { If there's no room left in the current section, create a new one }            if not TNewExeSection(CurrExeSec).CanAddObjSection(objsec,SegmentLimit) then              AddNewExeSection;            CurrExeSec.AddObjSection(objsec);          end;        TmpObjSectionList.Free;      end;    procedure TNewExeOutput.MemPos_Start;      var        i: Integer;      begin        inherited MemPos_Start;        for i:=0 to ExeSectionList.Count-1 do          begin            MemPos_ExeSection(TExeSection(ExeSectionList[i]));            CurrMemPos:=0;          end;      end;    procedure TNewExeOutput.GenerateLibraryImports(ImportLibraryList: TFPHashObjectList);      var        i,j: longint;        ImportLibrary: TImportLibrary;        ImportSymbol: TImportSymbol;        exesym: TExeSymbol;      begin        FImports:=ImportLibraryList;        AddImportLibrariesExtractedFromObjectModules;        for i:=0 to FImports.Count-1 do          begin            ImportLibrary:=TImportLibrary(FImports[i]);            for j:=0 to ImportLibrary.ImportSymbolList.Count-1 do              begin                ImportSymbol:=TImportSymbol(ImportLibrary.ImportSymbolList[j]);                exesym:=TExeSymbol(ExeSymbolList.Find(ImportSymbol.MangledName));                if assigned(exesym) and                   (exesym.State<>symstate_defined) then                  begin                    ImportSymbol.CachedExeSymbol:=exesym;                    exesym.State:=symstate_defined;                  end;              end;          end;        PackUnresolvedExeSymbols('after DLL imports');      end;    function TNewExeOutput.writeData: boolean;      begin        Result:=False;        if ExeWriteMode in [ewm_exefull,ewm_exeonly] then          begin            Result:=WriteNewExe;            if not Result then              exit;          end;      end;{****************************************************************************                               TOmfAssembler****************************************************************************}    constructor TOmfAssembler.Create(info: pasminfo; smart:boolean);      begin        inherited;        CObjOutput:=TOmfObjOutput;        CInternalAr:=TOmfLibObjectWriter;      end;{*****************************************************************************                            Procedures and functions*****************************************************************************}    function StripDllExt(const DllName:TSymStr):TSymStr;      begin        if UpCase(ExtractFileExt(DllName))='.DLL' then          Result:=Copy(DllName,1,Length(DllName)-4)        else          Result:=DllName;      end;    function MaybeAddDllExt(const DllName: TSymStr): TSymStr;      begin        if ExtractFileExt(DllName)='' then          Result:=ChangeFileExt(DllName,'.dll')        else          Result:=DllName;      end;{*****************************************************************************                                  Initialize*****************************************************************************}{$ifdef i8086}    const       as_i8086_omf_info : tasminfo =          (            id     : as_i8086_omf;            idtxt  : 'OMF';            asmbin : '';            asmcmd : '';            supported_targets : [system_i8086_msdos,system_i8086_embedded,system_i8086_win16];            flags : [af_outputbinary,af_smartlink_sections];            labelprefix : '..@';            labelmaxlen : -1;            comment : '; ';            dollarsign: '$';          );{$endif i8086}initialization{$ifdef i8086}  RegisterAssembler(as_i8086_omf_info,TOmfAssembler);{$endif i8086}end.
 |