JclPeImage.pas 220 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940
  1. {**************************************************************************************************}
  2. { }
  3. { Project JEDI Code Library (JCL) }
  4. { }
  5. { The contents of this file are subject to the Mozilla Public License Version 1.1 (the "License"); }
  6. { you may not use this file except in compliance with the License. You may obtain a copy of the }
  7. { License at http://www.mozilla.org/MPL/ }
  8. { }
  9. { Software distributed under the License is distributed on an "AS IS" basis, WITHOUT WARRANTY OF }
  10. { ANY KIND, either express or implied. See the License for the specific language governing rights }
  11. { and limitations under the License. }
  12. { }
  13. { The Original Code is JclPeImage.pas. }
  14. { }
  15. { The Initial Developer of the Original Code is Petr Vones. Portions created by Petr Vones are }
  16. { Copyright (C) Petr Vones. All Rights Reserved. }
  17. { }
  18. { Contributor(s): }
  19. { Marcel van Brakel }
  20. { Robert Marquardt (marquardt) }
  21. { Uwe Schuster (uschuster) }
  22. { Matthias Thoma (mthoma) }
  23. { Petr Vones (pvones) }
  24. { Hallvard Vassbotn }
  25. { Jean-Fabien Connault (cycocrew) }
  26. { }
  27. {**************************************************************************************************}
  28. { }
  29. { This unit contains various classes and support routines to read the contents of portable }
  30. { executable (PE) files. You can use these classes to, for example examine the contents of the }
  31. { imports section of an executable. In addition the unit contains support for Borland specific }
  32. { structures and name unmangling. }
  33. { }
  34. {**************************************************************************************************}
  35. { }
  36. { Last modified: $Date:: $ }
  37. { Revision: $Rev:: $ }
  38. { Author: $Author:: $ }
  39. { }
  40. {**************************************************************************************************}
  41. unit JclPeImage;
  42. {$I jcl.inc}
  43. {$I windowsonly.inc}
  44. interface
  45. uses
  46. {$IFDEF UNITVERSIONING}
  47. JclUnitVersioning,
  48. {$ENDIF UNITVERSIONING}
  49. {$IFDEF HAS_UNITSCOPE}
  50. Winapi.Windows, System.Classes, System.SysUtils, System.TypInfo, System.Contnrs,
  51. {$ELSE ~HAS_UNITSCOPE}
  52. Windows, Classes, SysUtils, TypInfo, Contnrs,
  53. {$ENDIF ~HAS_UNITSCOPE}
  54. JclBase, JclDateTime, JclFileUtils, JclWin32;
  55. type
  56. // Smart name compare function
  57. TJclSmartCompOption = (scSimpleCompare, scIgnoreCase);
  58. TJclSmartCompOptions = set of TJclSmartCompOption;
  59. function PeStripFunctionAW(const FunctionName: string): string;
  60. function PeSmartFunctionNameSame(const ComparedName, FunctionName: string;
  61. Options: TJclSmartCompOptions = []): Boolean;
  62. type
  63. // Base list
  64. EJclPeImageError = class(EJclError);
  65. TJclPeImage = class;
  66. TJclPeImageClass = class of TJclPeImage;
  67. TJclPeImageBaseList = class(TObjectList)
  68. private
  69. FImage: TJclPeImage;
  70. public
  71. constructor Create(AImage: TJclPeImage);
  72. property Image: TJclPeImage read FImage;
  73. end;
  74. // Images cache
  75. TJclPeImagesCache = class(TObject)
  76. private
  77. FList: TStringList;
  78. function GetCount: Integer;
  79. function GetImages(const FileName: TFileName): TJclPeImage;
  80. protected
  81. function GetPeImageClass: TJclPeImageClass; virtual;
  82. public
  83. constructor Create;
  84. destructor Destroy; override;
  85. procedure Clear;
  86. property Images[const FileName: TFileName]: TJclPeImage read GetImages; default;
  87. property Count: Integer read GetCount;
  88. end;
  89. // Import section related classes
  90. TJclPeImportSort = (isName, isOrdinal, isHint, isLibImport);
  91. TJclPeImportLibSort = (ilName, ilIndex);
  92. TJclPeImportKind = (ikImport, ikDelayImport, ikBoundImport);
  93. TJclPeResolveCheck = (icNotChecked, icResolved, icUnresolved);
  94. TJclPeLinkerProducer = (lrBorland, lrMicrosoft);
  95. // lrBorland -> Delphi PE files
  96. // lrMicrosoft -> MSVC and BCB PE files
  97. TJclPeImportLibItem = class;
  98. // Created from a IMAGE_THUNK_DATA64 or IMAGE_THUNK_DATA32 record
  99. TJclPeImportFuncItem = class(TObject)
  100. private
  101. FOrdinal: Word; // word in 32/64
  102. FHint: Word;
  103. FImportLib: TJclPeImportLibItem;
  104. FIndirectImportName: Boolean;
  105. FName: string;
  106. FResolveCheck: TJclPeResolveCheck;
  107. function GetIsByOrdinal: Boolean;
  108. protected
  109. procedure SetName(const Value: string);
  110. procedure SetIndirectImportName(const Value: string);
  111. procedure SetResolveCheck(Value: TJclPeResolveCheck);
  112. public
  113. constructor Create(AImportLib: TJclPeImportLibItem; AOrdinal: Word;
  114. AHint: Word; const AName: string);
  115. property Ordinal: Word read FOrdinal;
  116. property Hint: Word read FHint;
  117. property ImportLib: TJclPeImportLibItem read FImportLib;
  118. property IndirectImportName: Boolean read FIndirectImportName;
  119. property IsByOrdinal: Boolean read GetIsByOrdinal;
  120. property Name: string read FName;
  121. property ResolveCheck: TJclPeResolveCheck read FResolveCheck;
  122. end;
  123. // Created from a IMAGE_IMPORT_DESCRIPTOR
  124. TJclPeImportLibItem = class(TJclPeImageBaseList)
  125. private
  126. FImportDescriptor: Pointer;
  127. FImportDirectoryIndex: Integer;
  128. FImportKind: TJclPeImportKind;
  129. FLastSortType: TJclPeImportSort;
  130. FLastSortDescending: Boolean;
  131. FName: string;
  132. FSorted: Boolean;
  133. FTotalResolveCheck: TJclPeResolveCheck;
  134. FThunk: Pointer;
  135. FThunkData: Pointer;
  136. function GetCount: Integer;
  137. function GetFileName: TFileName;
  138. function GetItems(Index: Integer): TJclPeImportFuncItem;
  139. function GetName: string;
  140. function GetThunkData32: PImageThunkData32;
  141. function GetThunkData64: PImageThunkData64;
  142. protected
  143. procedure CheckImports(ExportImage: TJclPeImage);
  144. procedure CreateList;
  145. procedure SetImportDirectoryIndex(Value: Integer);
  146. procedure SetImportKind(Value: TJclPeImportKind);
  147. procedure SetSorted(Value: Boolean);
  148. procedure SetThunk(Value: Pointer);
  149. public
  150. constructor Create(AImage: TJclPeImage; AImportDescriptor: Pointer;
  151. AImportKind: TJclPeImportKind; const AName: string; AThunk: Pointer);
  152. procedure SortList(SortType: TJclPeImportSort; Descending: Boolean = False);
  153. property Count: Integer read GetCount;
  154. property FileName: TFileName read GetFileName;
  155. property ImportDescriptor: Pointer read FImportDescriptor;
  156. property ImportDirectoryIndex: Integer read FImportDirectoryIndex;
  157. property ImportKind: TJclPeImportKind read FImportKind;
  158. property Items[Index: Integer]: TJclPeImportFuncItem read GetItems; default;
  159. property Name: string read GetName;
  160. property OriginalName: string read FName;
  161. // use the following properties
  162. // property ThunkData: PImageThunkData
  163. property ThunkData32: PImageThunkData32 read GetThunkData32;
  164. property ThunkData64: PImageThunkData64 read GetThunkData64;
  165. property TotalResolveCheck: TJclPeResolveCheck read FTotalResolveCheck;
  166. end;
  167. TJclPeImportList = class(TJclPeImageBaseList)
  168. private
  169. FAllItemsList: TList;
  170. FFilterModuleName: string;
  171. FLastAllSortType: TJclPeImportSort;
  172. FLastAllSortDescending: Boolean;
  173. FLinkerProducer: TJclPeLinkerProducer;
  174. FParallelImportTable: array of Pointer;
  175. FUniqueNamesList: TStringList;
  176. function GetAllItemCount: Integer;
  177. function GetAllItems(Index: Integer): TJclPeImportFuncItem;
  178. function GetItems(Index: Integer): TJclPeImportLibItem;
  179. function GetUniqueLibItemCount: Integer;
  180. function GetUniqueLibItems(Index: Integer): TJclPeImportLibItem;
  181. function GetUniqueLibNames(Index: Integer): string;
  182. function GetUniqueLibItemFromName(const Name: string): TJclPeImportLibItem;
  183. procedure SetFilterModuleName(const Value: string);
  184. protected
  185. procedure CreateList;
  186. procedure RefreshAllItems;
  187. public
  188. constructor Create(AImage: TJclPeImage);
  189. destructor Destroy; override;
  190. procedure CheckImports(PeImageCache: TJclPeImagesCache = nil);
  191. function MakeBorlandImportTableForMappedImage: Boolean;
  192. function SmartFindName(const CompareName, LibName: string; Options: TJclSmartCompOptions = []): TJclPeImportFuncItem;
  193. procedure SortAllItemsList(SortType: TJclPeImportSort; Descending: Boolean = False);
  194. procedure SortList(SortType: TJclPeImportLibSort);
  195. procedure TryGetNamesForOrdinalImports;
  196. property AllItems[Index: Integer]: TJclPeImportFuncItem read GetAllItems;
  197. property AllItemCount: Integer read GetAllItemCount;
  198. property FilterModuleName: string read FFilterModuleName write SetFilterModuleName;
  199. property Items[Index: Integer]: TJclPeImportLibItem read GetItems; default;
  200. property LinkerProducer: TJclPeLinkerProducer read FLinkerProducer;
  201. property UniqueLibItemCount: Integer read GetUniqueLibItemCount;
  202. property UniqueLibItemFromName[const Name: string]: TJclPeImportLibItem read GetUniqueLibItemFromName;
  203. property UniqueLibItems[Index: Integer]: TJclPeImportLibItem read GetUniqueLibItems;
  204. property UniqueLibNames[Index: Integer]: string read GetUniqueLibNames;
  205. end;
  206. // Export section related classes
  207. TJclPeExportSort = (esName, esOrdinal, esHint, esAddress, esForwarded, esAddrOrFwd, esSection);
  208. TJclPeExportFuncList = class;
  209. // Created from a IMAGE_EXPORT_DIRECTORY
  210. TJclPeExportFuncItem = class(TObject)
  211. private
  212. FAddress: DWORD;
  213. FExportList: TJclPeExportFuncList;
  214. FForwardedName: string;
  215. FForwardedDotPos: string;
  216. FHint: Word;
  217. FName: string;
  218. FOrdinal: Word;
  219. FResolveCheck: TJclPeResolveCheck;
  220. function GetAddressOrForwardStr: string;
  221. function GetForwardedFuncName: string;
  222. function GetForwardedLibName: string;
  223. function GetForwardedFuncOrdinal: DWORD;
  224. function GetIsExportedVariable: Boolean;
  225. function GetIsForwarded: Boolean;
  226. function GetSectionName: string;
  227. function GetMappedAddress: Pointer;
  228. protected
  229. procedure SetResolveCheck(Value: TJclPeResolveCheck);
  230. public
  231. constructor Create(AExportList: TJclPeExportFuncList; const AName, AForwardedName: string;
  232. AAddress: DWORD; AHint: Word; AOrdinal: Word; AResolveCheck: TJclPeResolveCheck);
  233. property Address: DWORD read FAddress;
  234. property AddressOrForwardStr: string read GetAddressOrForwardStr;
  235. property IsExportedVariable: Boolean read GetIsExportedVariable;
  236. property IsForwarded: Boolean read GetIsForwarded;
  237. property ForwardedName: string read FForwardedName;
  238. property ForwardedLibName: string read GetForwardedLibName;
  239. property ForwardedFuncOrdinal: DWORD read GetForwardedFuncOrdinal;
  240. property ForwardedFuncName: string read GetForwardedFuncName;
  241. property Hint: Word read FHint;
  242. property MappedAddress: Pointer read GetMappedAddress;
  243. property Name: string read FName;
  244. property Ordinal: Word read FOrdinal;
  245. property ResolveCheck: TJclPeResolveCheck read FResolveCheck;
  246. property SectionName: string read GetSectionName;
  247. end;
  248. TJclPeExportFuncList = class(TJclPeImageBaseList)
  249. private
  250. FAnyForwards: Boolean;
  251. FBase: DWORD;
  252. FExportDir: PImageExportDirectory;
  253. FForwardedLibsList: TStringList;
  254. FFunctionCount: DWORD;
  255. FLastSortType: TJclPeExportSort;
  256. FLastSortDescending: Boolean;
  257. FSorted: Boolean;
  258. FTotalResolveCheck: TJclPeResolveCheck;
  259. function GetForwardedLibsList: TStrings;
  260. function GetItems(Index: Integer): TJclPeExportFuncItem;
  261. function GetItemFromAddress(Address: DWORD): TJclPeExportFuncItem;
  262. function GetItemFromOrdinal(Ordinal: DWORD): TJclPeExportFuncItem;
  263. function GetItemFromName(const Name: string): TJclPeExportFuncItem;
  264. function GetName: string;
  265. protected
  266. function CanPerformFastNameSearch: Boolean;
  267. procedure CreateList;
  268. property LastSortType: TJclPeExportSort read FLastSortType;
  269. property LastSortDescending: Boolean read FLastSortDescending;
  270. property Sorted: Boolean read FSorted;
  271. public
  272. constructor Create(AImage: TJclPeImage);
  273. destructor Destroy; override;
  274. procedure CheckForwards(PeImageCache: TJclPeImagesCache = nil);
  275. class function ItemName(Item: TJclPeExportFuncItem): string;
  276. function OrdinalValid(Ordinal: DWORD): Boolean;
  277. procedure PrepareForFastNameSearch;
  278. function SmartFindName(const CompareName: string; Options: TJclSmartCompOptions = []): TJclPeExportFuncItem;
  279. procedure SortList(SortType: TJclPeExportSort; Descending: Boolean = False);
  280. property AnyForwards: Boolean read FAnyForwards;
  281. property Base: DWORD read FBase;
  282. property ExportDir: PImageExportDirectory read FExportDir;
  283. property ForwardedLibsList: TStrings read GetForwardedLibsList;
  284. property FunctionCount: DWORD read FFunctionCount;
  285. property Items[Index: Integer]: TJclPeExportFuncItem read GetItems; default;
  286. property ItemFromAddress[Address: DWORD]: TJclPeExportFuncItem read GetItemFromAddress;
  287. property ItemFromName[const Name: string]: TJclPeExportFuncItem read GetItemFromName;
  288. property ItemFromOrdinal[Ordinal: DWORD]: TJclPeExportFuncItem read GetItemFromOrdinal;
  289. property Name: string read GetName;
  290. property TotalResolveCheck: TJclPeResolveCheck read FTotalResolveCheck;
  291. end;
  292. // Resource section related classes
  293. TJclPeResourceKind = (
  294. rtUnknown0,
  295. rtCursorEntry,
  296. rtBitmap,
  297. rtIconEntry,
  298. rtMenu,
  299. rtDialog,
  300. rtString,
  301. rtFontDir,
  302. rtFont,
  303. rtAccelerators,
  304. rtRCData,
  305. rtMessageTable,
  306. rtCursor,
  307. rtUnknown13,
  308. rtIcon,
  309. rtUnknown15,
  310. rtVersion,
  311. rtDlgInclude,
  312. rtUnknown18,
  313. rtPlugPlay,
  314. rtVxd,
  315. rtAniCursor,
  316. rtAniIcon,
  317. rtHmtl,
  318. rtManifest,
  319. rtUserDefined);
  320. TJclPeResourceList = class;
  321. TJclPeResourceItem = class;
  322. TJclPeResourceRawStream = class(TCustomMemoryStream)
  323. public
  324. constructor Create(AResourceItem: TJclPeResourceItem);
  325. function Write(const Buffer; Count: Longint): Longint; override;
  326. end;
  327. TJclPeResourceItem = class(TObject)
  328. private
  329. FEntry: PImageResourceDirectoryEntry;
  330. FImage: TJclPeImage;
  331. FList: TJclPeResourceList;
  332. FLevel: Byte;
  333. FParentItem: TJclPeResourceItem;
  334. FNameCache: string;
  335. function GetDataEntry: PImageResourceDataEntry;
  336. function GetIsDirectory: Boolean;
  337. function GetIsName: Boolean;
  338. function GetLangID: LANGID;
  339. function GetList: TJclPeResourceList;
  340. function GetName: string;
  341. function GetParameterName: string;
  342. function GetRawEntryData: Pointer;
  343. function GetRawEntryDataSize: Integer;
  344. function GetResourceType: TJclPeResourceKind;
  345. function GetResourceTypeStr: string;
  346. protected
  347. function OffsetToRawData(Ofs: DWORD): TJclAddr;
  348. function Level1Item: TJclPeResourceItem;
  349. function SubDirData: PImageResourceDirectory;
  350. public
  351. constructor Create(AImage: TJclPeImage; AParentItem: TJclPeResourceItem;
  352. AEntry: PImageResourceDirectoryEntry);
  353. destructor Destroy; override;
  354. function CompareName(AName: PChar): Boolean;
  355. property DataEntry: PImageResourceDataEntry read GetDataEntry;
  356. property Entry: PImageResourceDirectoryEntry read FEntry;
  357. property Image: TJclPeImage read FImage;
  358. property IsDirectory: Boolean read GetIsDirectory;
  359. property IsName: Boolean read GetIsName;
  360. property LangID: LANGID read GetLangID;
  361. property List: TJclPeResourceList read GetList;
  362. property Level: Byte read FLevel;
  363. property Name: string read GetName;
  364. property ParameterName: string read GetParameterName;
  365. property ParentItem: TJclPeResourceItem read FParentItem;
  366. property RawEntryData: Pointer read GetRawEntryData;
  367. property RawEntryDataSize: Integer read GetRawEntryDataSize;
  368. property ResourceType: TJclPeResourceKind read GetResourceType;
  369. property ResourceTypeStr: string read GetResourceTypeStr;
  370. end;
  371. TJclPeResourceList = class(TJclPeImageBaseList)
  372. private
  373. FDirectory: PImageResourceDirectory;
  374. FParentItem: TJclPeResourceItem;
  375. function GetItems(Index: Integer): TJclPeResourceItem;
  376. protected
  377. procedure CreateList(AParentItem: TJclPeResourceItem);
  378. public
  379. constructor Create(AImage: TJclPeImage; AParentItem: TJclPeResourceItem;
  380. ADirectory: PImageResourceDirectory);
  381. function FindName(const Name: string): TJclPeResourceItem;
  382. property Directory: PImageResourceDirectory read FDirectory;
  383. property Items[Index: Integer]: TJclPeResourceItem read GetItems; default;
  384. property ParentItem: TJclPeResourceItem read FParentItem;
  385. end;
  386. TJclPeRootResourceList = class(TJclPeResourceList)
  387. private
  388. FManifestContent: TStringList;
  389. function GetManifestContent: TStrings;
  390. public
  391. destructor Destroy; override;
  392. function FindResource(ResourceType: TJclPeResourceKind;
  393. const ResourceName: string = ''): TJclPeResourceItem; overload;
  394. function FindResource(const ResourceType: PChar;
  395. const ResourceName: PChar = nil): TJclPeResourceItem; overload;
  396. function ListResourceNames(ResourceType: TJclPeResourceKind; const Strings: TStrings): Boolean;
  397. property ManifestContent: TStrings read GetManifestContent;
  398. end;
  399. // Relocation section related classes
  400. TJclPeRelocation = record
  401. Address: Word;
  402. RelocType: Byte;
  403. VirtualAddress: DWORD;
  404. end;
  405. TJclPeRelocEntry = class(TObject)
  406. private
  407. FChunk: PImageBaseRelocation;
  408. FCount: Integer;
  409. function GetRelocations(Index: Integer): TJclPeRelocation;
  410. function GetSize: DWORD;
  411. function GetVirtualAddress: DWORD;
  412. public
  413. constructor Create(AChunk: PImageBaseRelocation; ACount: Integer);
  414. property Count: Integer read FCount;
  415. property Relocations[Index: Integer]: TJclPeRelocation read GetRelocations; default;
  416. property Size: DWORD read GetSize;
  417. property VirtualAddress: DWORD read GetVirtualAddress;
  418. end;
  419. TJclPeRelocList = class(TJclPeImageBaseList)
  420. private
  421. FAllItemCount: Integer;
  422. function GetItems(Index: Integer): TJclPeRelocEntry;
  423. function GetAllItems(Index: Integer): TJclPeRelocation;
  424. protected
  425. procedure CreateList;
  426. public
  427. constructor Create(AImage: TJclPeImage);
  428. property AllItems[Index: Integer]: TJclPeRelocation read GetAllItems;
  429. property AllItemCount: Integer read FAllItemCount;
  430. property Items[Index: Integer]: TJclPeRelocEntry read GetItems; default;
  431. end;
  432. // Debug section related classes
  433. TJclPeDebugList = class(TJclPeImageBaseList)
  434. private
  435. function GetItems(Index: Integer): TImageDebugDirectory;
  436. protected
  437. procedure CreateList;
  438. public
  439. constructor Create(AImage: TJclPeImage);
  440. property Items[Index: Integer]: TImageDebugDirectory read GetItems; default;
  441. end;
  442. // Certificates section related classes
  443. TJclPeCertificate = class(TObject)
  444. private
  445. FData: Pointer;
  446. FHeader: TWinCertificate;
  447. public
  448. constructor Create(AHeader: TWinCertificate; AData: Pointer);
  449. property Data: Pointer read FData;
  450. property Header: TWinCertificate read FHeader;
  451. end;
  452. TJclPeCertificateList = class(TJclPeImageBaseList)
  453. private
  454. function GetItems(Index: Integer): TJclPeCertificate;
  455. protected
  456. procedure CreateList;
  457. public
  458. constructor Create(AImage: TJclPeImage);
  459. property Items[Index: Integer]: TJclPeCertificate read GetItems; default;
  460. end;
  461. // Common Language Runtime section related classes
  462. TJclPeCLRHeader = class(TObject)
  463. private
  464. FHeader: TImageCor20Header;
  465. FImage: TJclPeImage;
  466. function GetVersionString: string;
  467. function GetHasMetadata: Boolean;
  468. protected
  469. procedure ReadHeader;
  470. public
  471. constructor Create(AImage: TJclPeImage);
  472. property HasMetadata: Boolean read GetHasMetadata;
  473. property Header: TImageCor20Header read FHeader;
  474. property VersionString: string read GetVersionString;
  475. property Image: TJclPeImage read FImage;
  476. end;
  477. // PE Image
  478. TJclPeHeader = (
  479. JclPeHeader_Signature,
  480. JclPeHeader_Machine,
  481. JclPeHeader_NumberOfSections,
  482. JclPeHeader_TimeDateStamp,
  483. JclPeHeader_PointerToSymbolTable,
  484. JclPeHeader_NumberOfSymbols,
  485. JclPeHeader_SizeOfOptionalHeader,
  486. JclPeHeader_Characteristics,
  487. JclPeHeader_Magic,
  488. JclPeHeader_LinkerVersion,
  489. JclPeHeader_SizeOfCode,
  490. JclPeHeader_SizeOfInitializedData,
  491. JclPeHeader_SizeOfUninitializedData,
  492. JclPeHeader_AddressOfEntryPoint,
  493. JclPeHeader_BaseOfCode,
  494. JclPeHeader_BaseOfData,
  495. JclPeHeader_ImageBase,
  496. JclPeHeader_SectionAlignment,
  497. JclPeHeader_FileAlignment,
  498. JclPeHeader_OperatingSystemVersion,
  499. JclPeHeader_ImageVersion,
  500. JclPeHeader_SubsystemVersion,
  501. JclPeHeader_Win32VersionValue,
  502. JclPeHeader_SizeOfImage,
  503. JclPeHeader_SizeOfHeaders,
  504. JclPeHeader_CheckSum,
  505. JclPeHeader_Subsystem,
  506. JclPeHeader_DllCharacteristics,
  507. JclPeHeader_SizeOfStackReserve,
  508. JclPeHeader_SizeOfStackCommit,
  509. JclPeHeader_SizeOfHeapReserve,
  510. JclPeHeader_SizeOfHeapCommit,
  511. JclPeHeader_LoaderFlags,
  512. JclPeHeader_NumberOfRvaAndSizes);
  513. TJclLoadConfig = (
  514. JclLoadConfig_Characteristics, { TODO : rename to Size? }
  515. JclLoadConfig_TimeDateStamp,
  516. JclLoadConfig_Version,
  517. JclLoadConfig_GlobalFlagsClear,
  518. JclLoadConfig_GlobalFlagsSet,
  519. JclLoadConfig_CriticalSectionDefaultTimeout,
  520. JclLoadConfig_DeCommitFreeBlockThreshold,
  521. JclLoadConfig_DeCommitTotalFreeThreshold,
  522. JclLoadConfig_LockPrefixTable,
  523. JclLoadConfig_MaximumAllocationSize,
  524. JclLoadConfig_VirtualMemoryThreshold,
  525. JclLoadConfig_ProcessHeapFlags,
  526. JclLoadConfig_ProcessAffinityMask,
  527. JclLoadConfig_CSDVersion,
  528. JclLoadConfig_Reserved1,
  529. JclLoadConfig_EditList,
  530. JclLoadConfig_Reserved { TODO : extend to the new fields? }
  531. );
  532. TJclPeFileProperties = record
  533. Size: DWORD;
  534. CreationTime: TDateTime;
  535. LastAccessTime: TDateTime;
  536. LastWriteTime: TDateTime;
  537. Attributes: Integer;
  538. end;
  539. TJclPeImageStatus = (stNotLoaded, stOk, stNotPE, stNotSupported, stNotFound, stError);
  540. TJclPeTarget = (taUnknown, taWin32, taWin64);
  541. TJclPeImage = class(TObject)
  542. private
  543. FAttachedImage: Boolean;
  544. FCertificateList: TJclPeCertificateList;
  545. FCLRHeader: TJclPeCLRHeader;
  546. FDebugList: TJclPeDebugList;
  547. FFileName: TFileName;
  548. FImageSections: TStringList;
  549. FLoadedImage: TLoadedImage;
  550. FExportList: TJclPeExportFuncList;
  551. FImportList: TJclPeImportList;
  552. FNoExceptions: Boolean;
  553. FReadOnlyAccess: Boolean;
  554. FRelocationList: TJclPeRelocList;
  555. FResourceList: TJclPeRootResourceList;
  556. FResourceVA: TJclAddr;
  557. FStatus: TJclPeImageStatus;
  558. FTarget: TJclPeTarget;
  559. FVersionInfo: TJclFileVersionInfo;
  560. FStringTable: TStringList;
  561. function GetCertificateList: TJclPeCertificateList;
  562. function GetCLRHeader: TJclPeCLRHeader;
  563. function GetDebugList: TJclPeDebugList;
  564. function GetDescription: string;
  565. function GetDirectories(Directory: Word): TImageDataDirectory;
  566. function GetDirectoryExists(Directory: Word): Boolean;
  567. function GetExportList: TJclPeExportFuncList;
  568. function GetFileProperties: TJclPeFileProperties;
  569. function GetImageSectionCount: Integer;
  570. function GetImageSectionHeaders(Index: Integer): TImageSectionHeader;
  571. function GetImageSectionNames(Index: Integer): string;
  572. function GetImageSectionNameFromRva(const Rva: DWORD): string;
  573. function GetImportList: TJclPeImportList;
  574. function GetHeaderValues(Index: TJclPeHeader): string;
  575. function GetLoadConfigValues(Index: TJclLoadConfig): string;
  576. function GetMappedAddress: TJclAddr;
  577. function GetOptionalHeader32: TImageOptionalHeader32;
  578. function GetOptionalHeader64: TImageOptionalHeader64;
  579. function GetRelocationList: TJclPeRelocList;
  580. function GetResourceList: TJclPeRootResourceList;
  581. function GetUnusedHeaderBytes: TImageDataDirectory;
  582. function GetVersionInfo: TJclFileVersionInfo;
  583. function GetVersionInfoAvailable: Boolean;
  584. procedure ReadImageSections;
  585. procedure ReadStringTable;
  586. procedure SetFileName(const Value: TFileName);
  587. function GetStringTableCount: Integer;
  588. function GetStringTableItem(Index: Integer): string;
  589. function GetImageSectionFullNames(Index: Integer): string;
  590. protected
  591. procedure AfterOpen; dynamic;
  592. procedure CheckNotAttached;
  593. procedure Clear; dynamic;
  594. function ExpandModuleName(const ModuleName: string): TFileName;
  595. procedure RaiseStatusException;
  596. function ResourceItemCreate(AEntry: PImageResourceDirectoryEntry;
  597. AParentItem: TJclPeResourceItem): TJclPeResourceItem; virtual;
  598. function ResourceListCreate(ADirectory: PImageResourceDirectory;
  599. AParentItem: TJclPeResourceItem): TJclPeResourceList; virtual;
  600. property NoExceptions: Boolean read FNoExceptions;
  601. public
  602. constructor Create(ANoExceptions: Boolean = False); virtual;
  603. destructor Destroy; override;
  604. procedure AttachLoadedModule(const Handle: HMODULE);
  605. function CalculateCheckSum: DWORD;
  606. function DirectoryEntryToData(Directory: Word): Pointer;
  607. function GetSectionHeader(const SectionName: string; out Header: PImageSectionHeader): Boolean;
  608. function GetSectionName(Header: PImageSectionHeader): string;
  609. function GetNameInStringTable(Offset: ULONG): string;
  610. function IsBrokenFormat: Boolean;
  611. function IsCLR: Boolean;
  612. function IsSystemImage: Boolean;
  613. // RVA are always DWORD
  614. function RawToVa(Raw: DWORD): Pointer; overload;
  615. function RvaToSection(Rva: DWORD): PImageSectionHeader; overload;
  616. function RvaToVa(Rva: DWORD): Pointer; overload;
  617. function RvaToVaEx(Rva: DWORD): Pointer; overload;
  618. function StatusOK: Boolean;
  619. procedure TryGetNamesForOrdinalImports;
  620. function VerifyCheckSum: Boolean;
  621. class function DebugTypeNames(DebugType: DWORD): string;
  622. class function DirectoryNames(Directory: Word): string;
  623. class function ExpandBySearchPath(const ModuleName, BasePath: string): TFileName;
  624. class function HeaderNames(Index: TJclPeHeader): string;
  625. class function LoadConfigNames(Index: TJclLoadConfig): string;
  626. class function ShortSectionInfo(Characteristics: DWORD): string;
  627. class function DateTimeToStamp(const DateTime: TDateTime): DWORD;
  628. class function StampToDateTime(TimeDateStamp: DWORD): TDateTime;
  629. property AttachedImage: Boolean read FAttachedImage;
  630. property CertificateList: TJclPeCertificateList read GetCertificateList;
  631. property CLRHeader: TJclPeCLRHeader read GetCLRHeader;
  632. property DebugList: TJclPeDebugList read GetDebugList;
  633. property Description: string read GetDescription;
  634. property Directories[Directory: Word]: TImageDataDirectory read GetDirectories;
  635. property DirectoryExists[Directory: Word]: Boolean read GetDirectoryExists;
  636. property ExportList: TJclPeExportFuncList read GetExportList;
  637. property FileName: TFileName read FFileName write SetFileName;
  638. property FileProperties: TJclPeFileProperties read GetFileProperties;
  639. property HeaderValues[Index: TJclPeHeader]: string read GetHeaderValues;
  640. property ImageSectionCount: Integer read GetImageSectionCount;
  641. property ImageSectionHeaders[Index: Integer]: TImageSectionHeader read GetImageSectionHeaders;
  642. property ImageSectionNames[Index: Integer]: string read GetImageSectionNames;
  643. property ImageSectionFullNames[Index: Integer]: string read GetImageSectionFullNames;
  644. property ImageSectionNameFromRva[const Rva: DWORD]: string read GetImageSectionNameFromRva;
  645. property ImportList: TJclPeImportList read GetImportList;
  646. property LoadConfigValues[Index: TJclLoadConfig]: string read GetLoadConfigValues;
  647. property LoadedImage: TLoadedImage read FLoadedImage;
  648. property MappedAddress: TJclAddr read GetMappedAddress;
  649. property StringTableCount: Integer read GetStringTableCount;
  650. property StringTable[Index: Integer]: string read GetStringTableItem;
  651. // use the following properties
  652. // property OptionalHeader: TImageOptionalHeader
  653. property OptionalHeader32: TImageOptionalHeader32 read GetOptionalHeader32;
  654. property OptionalHeader64: TImageOptionalHeader64 read GetOptionalHeader64;
  655. property ReadOnlyAccess: Boolean read FReadOnlyAccess write FReadOnlyAccess;
  656. property RelocationList: TJclPeRelocList read GetRelocationList;
  657. property ResourceVA: TJclAddr read FResourceVA;
  658. property ResourceList: TJclPeRootResourceList read GetResourceList;
  659. property Status: TJclPeImageStatus read FStatus;
  660. property Target: TJclPeTarget read FTarget;
  661. property UnusedHeaderBytes: TImageDataDirectory read GetUnusedHeaderBytes;
  662. property VersionInfo: TJclFileVersionInfo read GetVersionInfo;
  663. property VersionInfoAvailable: Boolean read GetVersionInfoAvailable;
  664. end;
  665. {$IFDEF BORLAND}
  666. TJclPeBorImage = class;
  667. TJclPeBorImagesCache = class(TJclPeImagesCache)
  668. private
  669. function GetImages(const FileName: TFileName): TJclPeBorImage;
  670. protected
  671. function GetPeImageClass: TJclPeImageClass; override;
  672. public
  673. property Images[const FileName: TFileName]: TJclPeBorImage read GetImages; default;
  674. end;
  675. // Borland Delphi PE Image specific information
  676. TJclPePackageInfo = class(TObject)
  677. private
  678. FAvailable: Boolean;
  679. FContains: TStringList;
  680. FDcpName: string;
  681. FRequires: TStringList;
  682. FFlags: Integer;
  683. FDescription: string;
  684. FEnsureExtension: Boolean;
  685. FSorted: Boolean;
  686. function GetContains: TStrings;
  687. function GetContainsCount: Integer;
  688. function GetContainsFlags(Index: Integer): Byte;
  689. function GetContainsNames(Index: Integer): string;
  690. function GetRequires: TStrings;
  691. function GetRequiresCount: Integer;
  692. function GetRequiresNames(Index: Integer): string;
  693. protected
  694. procedure ReadPackageInfo(ALibHandle: THandle);
  695. procedure SetDcpName(const Value: string);
  696. public
  697. constructor Create(ALibHandle: THandle);
  698. destructor Destroy; override;
  699. class function PackageModuleTypeToString(Flags: Cardinal): string;
  700. class function PackageOptionsToString(Flags: Cardinal): string;
  701. class function ProducerToString(Flags: Cardinal): string;
  702. class function UnitInfoFlagsToString(UnitFlags: Byte): string;
  703. property Available: Boolean read FAvailable;
  704. property Contains: TStrings read GetContains;
  705. property ContainsCount: Integer read GetContainsCount;
  706. property ContainsNames[Index: Integer]: string read GetContainsNames;
  707. property ContainsFlags[Index: Integer]: Byte read GetContainsFlags;
  708. property Description: string read FDescription;
  709. property DcpName: string read FDcpName;
  710. property EnsureExtension: Boolean read FEnsureExtension write FEnsureExtension;
  711. property Flags: Integer read FFlags;
  712. property Requires: TStrings read GetRequires;
  713. property RequiresCount: Integer read GetRequiresCount;
  714. property RequiresNames[Index: Integer]: string read GetRequiresNames;
  715. property Sorted: Boolean read FSorted write FSorted;
  716. end;
  717. TJclPeBorForm = class(TObject)
  718. private
  719. FFormFlags: TFilerFlags;
  720. FFormClassName: string;
  721. FFormObjectName: string;
  722. FFormPosition: Integer;
  723. FResItem: TJclPeResourceItem;
  724. function GetDisplayName: string;
  725. public
  726. constructor Create(AResItem: TJclPeResourceItem; AFormFlags: TFilerFlags;
  727. AFormPosition: Integer; const AFormClassName, AFormObjectName: string);
  728. procedure ConvertFormToText(const Stream: TStream); overload;
  729. procedure ConvertFormToText(const Strings: TStrings); overload;
  730. property FormClassName: string read FFormClassName;
  731. property FormFlags: TFilerFlags read FFormFlags;
  732. property FormObjectName: string read FFormObjectName;
  733. property FormPosition: Integer read FFormPosition;
  734. property DisplayName: string read GetDisplayName;
  735. property ResItem: TJclPeResourceItem read FResItem;
  736. end;
  737. TJclPeBorImage = class(TJclPeImage)
  738. private
  739. FForms: TObjectList;
  740. FIsPackage: Boolean;
  741. FIsBorlandImage: Boolean;
  742. FLibHandle: THandle;
  743. FPackageInfo: TJclPePackageInfo;
  744. FPackageInfoSorted: Boolean;
  745. FPackageCompilerVersion: Integer;
  746. function GetFormCount: Integer;
  747. function GetForms(Index: Integer): TJclPeBorForm;
  748. function GetFormFromName(const FormClassName: string): TJclPeBorForm;
  749. function GetLibHandle: THandle;
  750. function GetPackageCompilerVersion: Integer;
  751. function GetPackageInfo: TJclPePackageInfo;
  752. protected
  753. procedure AfterOpen; override;
  754. procedure Clear; override;
  755. procedure CreateFormsList;
  756. public
  757. constructor Create(ANoExceptions: Boolean = False); override;
  758. destructor Destroy; override;
  759. function DependedPackages(List: TStrings; FullPathName, Descriptions: Boolean): Boolean;
  760. function FreeLibHandle: Boolean;
  761. property Forms[Index: Integer]: TJclPeBorForm read GetForms;
  762. property FormCount: Integer read GetFormCount;
  763. property FormFromName[const FormClassName: string]: TJclPeBorForm read GetFormFromName;
  764. property IsBorlandImage: Boolean read FIsBorlandImage;
  765. property IsPackage: Boolean read FIsPackage;
  766. property LibHandle: THandle read GetLibHandle;
  767. property PackageCompilerVersion: Integer read GetPackageCompilerVersion;
  768. property PackageInfo: TJclPePackageInfo read GetPackageInfo;
  769. property PackageInfoSorted: Boolean read FPackageInfoSorted write FPackageInfoSorted;
  770. end;
  771. {$ENDIF BORLAND}
  772. // Threaded function search
  773. TJclPeNameSearchOption = (seImports, seDelayImports, seBoundImports, seExports);
  774. TJclPeNameSearchOptions = set of TJclPeNameSearchOption;
  775. TJclPeNameSearchNotifyEvent = procedure (Sender: TObject; PeImage: TJclPeImage;
  776. var Process: Boolean) of object;
  777. TJclPeNameSearchFoundEvent = procedure (Sender: TObject; const FileName: TFileName;
  778. const FunctionName: string; Option: TJclPeNameSearchOption) of object;
  779. TJclPeNameSearch = class(TThread)
  780. private
  781. F_FileName: TFileName;
  782. F_FunctionName: string;
  783. F_Option: TJclPeNameSearchOption;
  784. F_Process: Boolean;
  785. FFunctionName: string;
  786. FOptions: TJclPeNameSearchOptions;
  787. FPath: string;
  788. FPeImage: TJclPeImage;
  789. FOnFound: TJclPeNameSearchFoundEvent;
  790. FOnProcessFile: TJclPeNameSearchNotifyEvent;
  791. protected
  792. function CompareName(const FunctionName, ComparedName: string): Boolean; virtual;
  793. procedure DoFound;
  794. procedure DoProcessFile;
  795. procedure Execute; override;
  796. public
  797. constructor Create(const FunctionName, Path: string; Options: TJclPeNameSearchOptions = [seImports, seExports]);
  798. procedure Start;
  799. property OnFound: TJclPeNameSearchFoundEvent read FOnFound write FOnFound;
  800. property OnProcessFile: TJclPeNameSearchNotifyEvent read FOnProcessFile write FOnProcessFile;
  801. end;
  802. // PE Image miscellaneous functions
  803. type
  804. TJclRebaseImageInfo32 = record
  805. OldImageSize: DWORD;
  806. OldImageBase: TJclAddr32;
  807. NewImageSize: DWORD;
  808. NewImageBase: TJclAddr32;
  809. end;
  810. TJclRebaseImageInfo64 = record
  811. OldImageSize: DWORD;
  812. OldImageBase: TJclAddr64;
  813. NewImageSize: DWORD;
  814. NewImageBase: TJclAddr64;
  815. end;
  816. // renamed
  817. // TJclRebaseImageInfo = TJclRebaseImageInfo32;
  818. { Image validity }
  819. function IsValidPeFile(const FileName: TFileName): Boolean;
  820. // use PeGetNtHeaders32 for backward compatibility
  821. // function PeGetNtHeaders(const FileName: TFileName; out NtHeaders: TImageNtHeaders): Boolean;
  822. function PeGetNtHeaders32(const FileName: TFileName; out NtHeaders: TImageNtHeaders32): Boolean;
  823. function PeGetNtHeaders64(const FileName: TFileName; out NtHeaders: TImageNtHeaders64): Boolean;
  824. { Image modifications }
  825. function PeCreateNameHintTable(const FileName: TFileName): Boolean;
  826. // use PeRebaseImage32
  827. //function PeRebaseImage(const ImageName: TFileName; NewBase: DWORD = 0; TimeStamp: DWORD = 0;
  828. // MaxNewSize: DWORD = 0): TJclRebaseImageInfo;
  829. function PeRebaseImage32(const ImageName: TFileName; NewBase: TJclAddr32 = 0; TimeStamp: DWORD = 0;
  830. MaxNewSize: DWORD = 0): TJclRebaseImageInfo32;
  831. function PeRebaseImage64(const ImageName: TFileName; NewBase: TJclAddr64 = 0; TimeStamp: DWORD = 0;
  832. MaxNewSize: DWORD = 0): TJclRebaseImageInfo64;
  833. function PeUpdateLinkerTimeStamp(const FileName: TFileName; const Time: TDateTime): Boolean;
  834. function PeReadLinkerTimeStamp(const FileName: TFileName): TDateTime;
  835. function PeInsertSection(const FileName: TFileName; SectionStream: TStream; SectionName: string): Boolean;
  836. { Image Checksum }
  837. function PeVerifyCheckSum(const FileName: TFileName): Boolean;
  838. function PeClearCheckSum(const FileName: TFileName): Boolean;
  839. function PeUpdateCheckSum(const FileName: TFileName): Boolean;
  840. // Various simple PE Image searching and listing routines
  841. { Exports searching }
  842. function PeDoesExportFunction(const FileName: TFileName; const FunctionName: string;
  843. Options: TJclSmartCompOptions = []): Boolean;
  844. function PeIsExportFunctionForwardedEx(const FileName: TFileName; const FunctionName: string;
  845. out ForwardedName: string; Options: TJclSmartCompOptions = []): Boolean;
  846. function PeIsExportFunctionForwarded(const FileName: TFileName; const FunctionName: string;
  847. Options: TJclSmartCompOptions = []): Boolean;
  848. { Imports searching }
  849. function PeDoesImportFunction(const FileName: TFileName; const FunctionName: string;
  850. const LibraryName: string = ''; Options: TJclSmartCompOptions = []): Boolean;
  851. function PeDoesImportLibrary(const FileName: TFileName; const LibraryName: string;
  852. Recursive: Boolean = False): Boolean;
  853. { Imports listing }
  854. function PeImportedLibraries(const FileName: TFileName; const LibrariesList: TStrings;
  855. Recursive: Boolean = False; FullPathName: Boolean = False): Boolean;
  856. function PeImportedFunctions(const FileName: TFileName; const FunctionsList: TStrings;
  857. const LibraryName: string = ''; IncludeLibNames: Boolean = False): Boolean;
  858. { Exports listing }
  859. function PeExportedFunctions(const FileName: TFileName; const FunctionsList: TStrings): Boolean;
  860. function PeExportedNames(const FileName: TFileName; const FunctionsList: TStrings): Boolean;
  861. function PeExportedVariables(const FileName: TFileName; const FunctionsList: TStrings): Boolean;
  862. { Resources listing }
  863. function PeResourceKindNames(const FileName: TFileName; ResourceType: TJclPeResourceKind;
  864. const NamesList: TStrings): Boolean;
  865. { Borland packages specific }
  866. {$IFDEF BORLAND}
  867. function PeBorFormNames(const FileName: TFileName; const NamesList: TStrings): Boolean;
  868. function PeBorDependedPackages(const FileName: TFileName; PackagesList: TStrings;
  869. FullPathName, Descriptions: Boolean): Boolean;
  870. {$ENDIF BORLAND}
  871. // Missing imports checking routines
  872. function PeFindMissingImports(const FileName: TFileName; MissingImportsList: TStrings): Boolean; overload;
  873. function PeFindMissingImports(RequiredImportsList, MissingImportsList: TStrings): Boolean; overload;
  874. function PeCreateRequiredImportList(const FileName: TFileName; RequiredImportsList: TStrings): Boolean;
  875. // Mapped or loaded image related routines
  876. // use PeMapImgNtHeaders32
  877. // function PeMapImgNtHeaders(const BaseAddress: Pointer): PImageNtHeaders;
  878. function PeMapImgNtHeaders32(const BaseAddress: Pointer): PImageNtHeaders32; overload;
  879. function PeMapImgNtHeaders32(Stream: TStream; const BasePosition: Int64; out NtHeaders32: TImageNtHeaders32): Int64; overload;
  880. function PeMapImgNtHeaders64(const BaseAddress: Pointer): PImageNtHeaders64; overload;
  881. function PeMapImgNtHeaders64(Stream: TStream; const BasePosition: Int64; out NtHeaders64: TImageNtHeaders64): Int64; overload;
  882. function PeMapImgLibraryName(const BaseAddress: Pointer): string;
  883. function PeMapImgLibraryName32(const BaseAddress: Pointer): string;
  884. function PeMapImgLibraryName64(const BaseAddress: Pointer): string;
  885. function PeMapImgSize(const BaseAddress: Pointer): DWORD; overload;
  886. function PeMapImgSize(Stream: TStream; const BasePosition: Int64): DWORD; overload;
  887. function PeMapImgSize32(const BaseAddress: Pointer): DWORD; overload;
  888. function PeMapImgSize32(Stream: TStream; const BasePosition: Int64): DWORD; overload;
  889. function PeMapImgSize64(const BaseAddress: Pointer): DWORD; overload;
  890. function PeMapImgSize64(Stream: TStream; const BasePosition: Int64): DWORD; overload;
  891. function PeMapImgTarget(const BaseAddress: Pointer): TJclPeTarget; overload;
  892. function PeMapImgTarget(Stream: TStream; const BasePosition: Int64): TJclPeTarget; overload;
  893. type
  894. TImageSectionHeaderArray = array of TImageSectionHeader;
  895. // use PeMapImgSections32
  896. // function PeMapImgSections(NtHeaders: PImageNtHeaders): PImageSectionHeader;
  897. function PeMapImgSections32(NtHeaders: PImageNtHeaders32): PImageSectionHeader; overload;
  898. function PeMapImgSections32(Stream: TStream; const NtHeaders32Position: Int64; const NtHeaders32: TImageNtHeaders32;
  899. out ImageSectionHeaders: TImageSectionHeaderArray): Int64; overload;
  900. function PeMapImgSections64(NtHeaders: PImageNtHeaders64): PImageSectionHeader; overload;
  901. function PeMapImgSections64(Stream: TStream; const NtHeaders64Position: Int64; const NtHeaders64: TImageNtHeaders64;
  902. out ImageSectionHeaders: TImageSectionHeaderArray): Int64; overload;
  903. // use PeMapImgFindSection32
  904. // function PeMapImgFindSection(NtHeaders: PImageNtHeaders;
  905. // const SectionName: string): PImageSectionHeader;
  906. function PeMapImgFindSection32(NtHeaders: PImageNtHeaders32;
  907. const SectionName: string): PImageSectionHeader;
  908. function PeMapImgFindSection64(NtHeaders: PImageNtHeaders64;
  909. const SectionName: string): PImageSectionHeader;
  910. function PeMapImgFindSection(const ImageSectionHeaders: TImageSectionHeaderArray;
  911. const SectionName: string): SizeInt;
  912. function PeMapImgFindSectionFromModule(const BaseAddress: Pointer;
  913. const SectionName: string): PImageSectionHeader;
  914. function PeMapImgExportedVariables(const Module: HMODULE; const VariablesList: TStrings): Boolean;
  915. function PeMapImgResolvePackageThunk(Address: Pointer): Pointer;
  916. function PeMapFindResource(const Module: HMODULE; const ResourceType: PChar;
  917. const ResourceName: string): Pointer;
  918. type
  919. TJclPeSectionStream = class(TCustomMemoryStream)
  920. private
  921. FInstance: HMODULE;
  922. FSectionHeader: TImageSectionHeader;
  923. procedure Initialize(Instance: HMODULE; const ASectionName: string);
  924. public
  925. constructor Create(Instance: HMODULE; const ASectionName: string);
  926. function Write(const Buffer; Count: Longint): Longint; override;
  927. property Instance: HMODULE read FInstance;
  928. property SectionHeader: TImageSectionHeader read FSectionHeader;
  929. end;
  930. // API hooking classes
  931. type
  932. TJclPeMapImgHookItem = class(TObject)
  933. private
  934. FBaseAddress: Pointer;
  935. FFunctionName: string;
  936. FModuleName: string;
  937. FNewAddress: Pointer;
  938. FOriginalAddress: Pointer;
  939. FList: TObjectList;
  940. protected
  941. function InternalUnhook: Boolean;
  942. public
  943. constructor Create(AList: TObjectList; const AFunctionName: string;
  944. const AModuleName: string; ABaseAddress, ANewAddress, AOriginalAddress: Pointer);
  945. destructor Destroy; override;
  946. function Unhook: Boolean;
  947. property BaseAddress: Pointer read FBaseAddress;
  948. property FunctionName: string read FFunctionName;
  949. property ModuleName: string read FModuleName;
  950. property NewAddress: Pointer read FNewAddress;
  951. property OriginalAddress: Pointer read FOriginalAddress;
  952. end;
  953. TJclPeMapImgHooks = class(TObjectList)
  954. private
  955. function GetItems(Index: Integer): TJclPeMapImgHookItem;
  956. function GetItemFromOriginalAddress(OriginalAddress: Pointer): TJclPeMapImgHookItem;
  957. function GetItemFromNewAddress(NewAddress: Pointer): TJclPeMapImgHookItem;
  958. public
  959. function HookImport(Base: Pointer; const ModuleName: string;
  960. const FunctionName: string; NewAddress: Pointer; var OriginalAddress: Pointer): Boolean;
  961. class function IsWin9xDebugThunk(P: Pointer): Boolean;
  962. class function ReplaceImport(Base: Pointer; const ModuleName: string; FromProc, ToProc: Pointer): Boolean;
  963. class function SystemBase: Pointer;
  964. procedure UnhookAll;
  965. function UnhookByNewAddress(NewAddress: Pointer): Boolean;
  966. procedure UnhookByBaseAddress(BaseAddress: Pointer);
  967. property Items[Index: Integer]: TJclPeMapImgHookItem read GetItems; default;
  968. property ItemFromOriginalAddress[OriginalAddress: Pointer]: TJclPeMapImgHookItem read GetItemFromOriginalAddress;
  969. property ItemFromNewAddress[NewAddress: Pointer]: TJclPeMapImgHookItem read GetItemFromNewAddress;
  970. end;
  971. // Image access under a debbuger
  972. function PeDbgImgNtHeaders32(ProcessHandle: THandle; BaseAddress: TJclAddr32;
  973. var NtHeaders: TImageNtHeaders32): Boolean;
  974. // TODO 64 bit version
  975. //function PeDbgImgNtHeaders64(ProcessHandle: THandle; BaseAddress: TJclAddr64;
  976. // var NtHeaders: TImageNtHeaders64): Boolean;
  977. function PeDbgImgLibraryName32(ProcessHandle: THandle; BaseAddress: TJclAddr32;
  978. var Name: string): Boolean;
  979. //function PeDbgImgLibraryName64(ProcessHandle: THandle; BaseAddress: TJclAddr64;
  980. // var Name: string): Boolean;
  981. // Borland BPL packages name unmangling
  982. type
  983. TJclBorUmSymbolKind = (skData, skFunction, skConstructor, skDestructor, skRTTI, skVTable);
  984. TJclBorUmSymbolModifier = (smQualified, smLinkProc);
  985. TJclBorUmSymbolModifiers = set of TJclBorUmSymbolModifier;
  986. TJclBorUmDescription = record
  987. Kind: TJclBorUmSymbolKind;
  988. Modifiers: TJclBorUmSymbolModifiers;
  989. end;
  990. TJclBorUmResult = (urOk, urNotMangled, urMicrosoft, urError);
  991. TJclPeUmResult = (umNotMangled, umBorland, umMicrosoft);
  992. function PeBorUnmangleName(const Name: string; out Unmangled: string;
  993. out Description: TJclBorUmDescription; out BasePos: Integer): TJclBorUmResult; overload;
  994. function PeBorUnmangleName(const Name: string; out Unmangled: string;
  995. out Description: TJclBorUmDescription): TJclBorUmResult; overload;
  996. function PeBorUnmangleName(const Name: string; out Unmangled: string): TJclBorUmResult; overload;
  997. function PeBorUnmangleName(const Name: string): string; overload;
  998. function PeIsNameMangled(const Name: string): TJclPeUmResult;
  999. function UndecorateSymbolName(const DecoratedName: string; out UnMangled: string; Flags: DWORD): Boolean;
  1000. function PeUnmangleName(const Name: string; out Unmangled: string): TJclPeUmResult;
  1001. {$IFDEF UNITVERSIONING}
  1002. const
  1003. UnitVersioning: TUnitVersionInfo = (
  1004. RCSfile: '$URL$';
  1005. Revision: '$Revision$';
  1006. Date: '$Date$';
  1007. LogPath: 'JCL\source\windows';
  1008. Extra: '';
  1009. Data: nil
  1010. );
  1011. {$ENDIF UNITVERSIONING}
  1012. implementation
  1013. uses
  1014. {$IFDEF HAS_UNITSCOPE}
  1015. System.RTLConsts,
  1016. System.Types, // for inlining TList.Remove
  1017. {$IFDEF HAS_UNIT_CHARACTER}
  1018. System.Character,
  1019. {$ENDIF HAS_UNIT_CHARACTER}
  1020. {$ELSE ~HAS_UNITSCOPE}
  1021. RTLConsts,
  1022. {$IFDEF HAS_UNIT_CHARACTER}
  1023. Character,
  1024. {$ENDIF HAS_UNIT_CHARACTER}
  1025. {$ENDIF ~HAS_UNITSCOPE}
  1026. JclLogic, JclResources, JclSysUtils, JclAnsiStrings, JclStrings, JclStringConversions;
  1027. const
  1028. MANIFESTExtension = '.manifest';
  1029. DebugSectionName = '.debug';
  1030. ReadOnlySectionName = '.rdata';
  1031. BinaryExtensionLibrary = '.dll';
  1032. {$IFDEF BORLAND}
  1033. CompilerExtensionDCP = '.dcp';
  1034. BinaryExtensionPackage = '.bpl';
  1035. PackageInfoResName = 'PACKAGEINFO';
  1036. DescriptionResName = 'DESCRIPTION';
  1037. PackageOptionsResName = 'PACKAGEOPTIONS';
  1038. DVclAlResName = 'DVCLAL';
  1039. {$ENDIF BORLAND}
  1040. // Helper routines
  1041. function AddFlagTextRes(var Text: string; const FlagText: PResStringRec; const Value, Mask: Cardinal): Boolean;
  1042. begin
  1043. Result := (Value and Mask <> 0);
  1044. if Result then
  1045. begin
  1046. if Length(Text) > 0 then
  1047. Text := Text + ', ';
  1048. Text := Text + LoadResString(FlagText);
  1049. end;
  1050. end;
  1051. function CompareResourceName(T1, T2: PChar): Boolean;
  1052. var
  1053. Long1, Long2: LongRec;
  1054. begin
  1055. {$IFDEF CPU64}
  1056. Long1 := LongRec(Int64Rec(T1).Lo);
  1057. Long2 := LongRec(Int64Rec(T2).Lo);
  1058. if (Int64Rec(T1).Hi = 0) and (Int64Rec(T2).Hi = 0) and (Long1.Hi = 0) and (Long2.Hi = 0) then
  1059. {$ENDIF CPU64}
  1060. {$IFDEF CPU32}
  1061. Long1 := LongRec(T1);
  1062. Long2 := LongRec(T2);
  1063. if (Long1.Hi = 0) or (Long2.Hi = 0) then
  1064. {$ENDIF CPU32}
  1065. Result := Long1.Lo = Long2.Lo
  1066. else
  1067. Result := (StrIComp(T1, T2) = 0);
  1068. end;
  1069. function CreatePeImage(const FileName: TFileName): TJclPeImage;
  1070. begin
  1071. Result := TJclPeImage.Create(True);
  1072. Result.FileName := FileName;
  1073. end;
  1074. function InternalImportedLibraries(const FileName: TFileName;
  1075. Recursive, FullPathName: Boolean; ExternalCache: TJclPeImagesCache): TStringList;
  1076. var
  1077. Cache: TJclPeImagesCache;
  1078. procedure ProcessLibraries(const AFileName: TFileName);
  1079. var
  1080. I: Integer;
  1081. S: TFileName;
  1082. ImportLib: TJclPeImportLibItem;
  1083. begin
  1084. with Cache[AFileName].ImportList do
  1085. for I := 0 to Count - 1 do
  1086. begin
  1087. ImportLib := Items[I];
  1088. if FullPathName then
  1089. S := ImportLib.FileName
  1090. else
  1091. S := TFileName(ImportLib.Name);
  1092. if Result.IndexOf(S) = -1 then
  1093. begin
  1094. Result.Add(S);
  1095. if Recursive then
  1096. ProcessLibraries(ImportLib.FileName);
  1097. end;
  1098. end;
  1099. end;
  1100. begin
  1101. if ExternalCache = nil then
  1102. Cache := TJclPeImagesCache.Create
  1103. else
  1104. Cache := ExternalCache;
  1105. try
  1106. Result := TStringList.Create;
  1107. try
  1108. Result.Sorted := True;
  1109. Result.Duplicates := dupIgnore;
  1110. ProcessLibraries(FileName);
  1111. except
  1112. FreeAndNil(Result);
  1113. raise;
  1114. end;
  1115. finally
  1116. if ExternalCache = nil then
  1117. Cache.Free;
  1118. end;
  1119. end;
  1120. // Smart name compare function
  1121. function PeStripFunctionAW(const FunctionName: string): string;
  1122. var
  1123. L: Integer;
  1124. begin
  1125. Result := FunctionName;
  1126. L := Length(Result);
  1127. if (L > 1) then
  1128. case Result[L] of
  1129. 'A', 'W':
  1130. if CharIsValidIdentifierLetter(Result[L - 1]) then
  1131. Delete(Result, L, 1);
  1132. end;
  1133. end;
  1134. function PeSmartFunctionNameSame(const ComparedName, FunctionName: string;
  1135. Options: TJclSmartCompOptions): Boolean;
  1136. var
  1137. S: string;
  1138. begin
  1139. if scIgnoreCase in Options then
  1140. Result := CompareText(FunctionName, ComparedName) = 0
  1141. else
  1142. Result := (FunctionName = ComparedName);
  1143. if (not Result) and not (scSimpleCompare in Options) then
  1144. begin
  1145. if Length(FunctionName) > 0 then
  1146. begin
  1147. S := PeStripFunctionAW(FunctionName);
  1148. if scIgnoreCase in Options then
  1149. Result := CompareText(S, ComparedName) = 0
  1150. else
  1151. Result := (S = ComparedName);
  1152. end
  1153. else
  1154. Result := False;
  1155. end;
  1156. end;
  1157. //=== { TJclPeImagesCache } ==================================================
  1158. constructor TJclPeImagesCache.Create;
  1159. begin
  1160. inherited Create;
  1161. FList := TStringList.Create;
  1162. FList.Sorted := True;
  1163. FList.Duplicates := dupIgnore;
  1164. end;
  1165. destructor TJclPeImagesCache.Destroy;
  1166. begin
  1167. Clear;
  1168. FreeAndNil(FList);
  1169. inherited Destroy;
  1170. end;
  1171. procedure TJclPeImagesCache.Clear;
  1172. var
  1173. I: Integer;
  1174. begin
  1175. with FList do
  1176. for I := 0 to Count - 1 do
  1177. Objects[I].Free;
  1178. FList.Clear;
  1179. end;
  1180. function TJclPeImagesCache.GetCount: Integer;
  1181. begin
  1182. Result := FList.Count;
  1183. end;
  1184. function TJclPeImagesCache.GetImages(const FileName: TFileName): TJclPeImage;
  1185. var
  1186. I: Integer;
  1187. begin
  1188. I := FList.IndexOf(FileName);
  1189. if I = -1 then
  1190. begin
  1191. Result := GetPeImageClass.Create(True);
  1192. Result.FileName := FileName;
  1193. FList.AddObject(FileName, Result);
  1194. end
  1195. else
  1196. Result := TJclPeImage(FList.Objects[I]);
  1197. end;
  1198. function TJclPeImagesCache.GetPeImageClass: TJclPeImageClass;
  1199. begin
  1200. Result := TJclPeImage;
  1201. end;
  1202. //=== { TJclPeImageBaseList } ================================================
  1203. constructor TJclPeImageBaseList.Create(AImage: TJclPeImage);
  1204. begin
  1205. inherited Create(True);
  1206. FImage := AImage;
  1207. end;
  1208. // Import sort functions
  1209. function ImportSortByName(Item1, Item2: Pointer): Integer;
  1210. begin
  1211. Result := CompareStr(TJclPeImportFuncItem(Item1).Name, TJclPeImportFuncItem(Item2).Name);
  1212. if Result = 0 then
  1213. Result := CompareStr(TJclPeImportFuncItem(Item1).ImportLib.Name, TJclPeImportFuncItem(Item2).ImportLib.Name);
  1214. if Result = 0 then
  1215. Result := TJclPeImportFuncItem(Item1).Ordinal - TJclPeImportFuncItem(Item2).Ordinal;
  1216. end;
  1217. function ImportSortByNameDESC(Item1, Item2: Pointer): Integer;
  1218. begin
  1219. Result := ImportSortByName(Item2, Item1);
  1220. end;
  1221. function ImportSortByHint(Item1, Item2: Pointer): Integer;
  1222. begin
  1223. Result := TJclPeImportFuncItem(Item1).Hint - TJclPeImportFuncItem(Item2).Hint;
  1224. end;
  1225. function ImportSortByHintDESC(Item1, Item2: Pointer): Integer;
  1226. begin
  1227. Result := ImportSortByHint(Item2, Item1);
  1228. end;
  1229. function ImportSortByDll(Item1, Item2: Pointer): Integer;
  1230. begin
  1231. Result := CompareStr(TJclPeImportFuncItem(Item1).ImportLib.Name,
  1232. TJclPeImportFuncItem(Item2).ImportLib.Name);
  1233. if Result = 0 then
  1234. Result := ImportSortByName(Item1, Item2);
  1235. end;
  1236. function ImportSortByDllDESC(Item1, Item2: Pointer): Integer;
  1237. begin
  1238. Result := ImportSortByDll(Item2, Item1);
  1239. end;
  1240. function ImportSortByOrdinal(Item1, Item2: Pointer): Integer;
  1241. begin
  1242. Result := CompareStr(TJclPeImportFuncItem(Item1).ImportLib.Name,
  1243. TJclPeImportFuncItem(Item2).ImportLib.Name);
  1244. if Result = 0 then
  1245. Result := TJclPeImportFuncItem(Item1).Ordinal - TJclPeImportFuncItem(Item2).Ordinal;
  1246. end;
  1247. function ImportSortByOrdinalDESC(Item1, Item2: Pointer): Integer;
  1248. begin
  1249. Result := ImportSortByOrdinal(Item2, Item1);
  1250. end;
  1251. function GetImportSortFunction(SortType: TJclPeImportSort; Descending: Boolean): TListSortCompare;
  1252. const
  1253. SortFunctions: array [TJclPeImportSort, Boolean] of TListSortCompare =
  1254. ((ImportSortByName, ImportSortByNameDESC),
  1255. (ImportSortByOrdinal, ImportSortByOrdinalDESC),
  1256. (ImportSortByHint, ImportSortByHintDESC),
  1257. (ImportSortByDll, ImportSortByDllDESC)
  1258. );
  1259. begin
  1260. Result := SortFunctions[SortType, Descending];
  1261. end;
  1262. function ImportLibSortByIndex(Item1, Item2: Pointer): Integer;
  1263. begin
  1264. Result := TJclPeImportLibItem(Item1).ImportDirectoryIndex -
  1265. TJclPeImportLibItem(Item2).ImportDirectoryIndex;
  1266. end;
  1267. function ImportLibSortByName(Item1, Item2: Pointer): Integer;
  1268. begin
  1269. Result := AnsiCompareStr(TJclPeImportLibItem(Item1).Name, TJclPeImportLibItem(Item2).Name);
  1270. if Result = 0 then
  1271. Result := ImportLibSortByIndex(Item1, Item2);
  1272. end;
  1273. function GetImportLibSortFunction(SortType: TJclPeImportLibSort): TListSortCompare;
  1274. const
  1275. SortFunctions: array [TJclPeImportLibSort] of TListSortCompare =
  1276. (ImportLibSortByName, ImportLibSortByIndex);
  1277. begin
  1278. Result := SortFunctions[SortType];
  1279. end;
  1280. //=== { TJclPeImportFuncItem } ===============================================
  1281. constructor TJclPeImportFuncItem.Create(AImportLib: TJclPeImportLibItem;
  1282. AOrdinal: Word; AHint: Word; const AName: string);
  1283. begin
  1284. inherited Create;
  1285. FImportLib := AImportLib;
  1286. FOrdinal := AOrdinal;
  1287. FHint := AHint;
  1288. FName := AName;
  1289. FResolveCheck := icNotChecked;
  1290. FIndirectImportName := False;
  1291. end;
  1292. function TJclPeImportFuncItem.GetIsByOrdinal: Boolean;
  1293. begin
  1294. Result := FOrdinal <> 0;
  1295. end;
  1296. procedure TJclPeImportFuncItem.SetIndirectImportName(const Value: string);
  1297. begin
  1298. FName := Value;
  1299. FIndirectImportName := True;
  1300. end;
  1301. procedure TJclPeImportFuncItem.SetName(const Value: string);
  1302. begin
  1303. FName := Value;
  1304. FIndirectImportName := False;
  1305. end;
  1306. procedure TJclPeImportFuncItem.SetResolveCheck(Value: TJclPeResolveCheck);
  1307. begin
  1308. FResolveCheck := Value;
  1309. end;
  1310. //=== { TJclPeImportLibItem } ================================================
  1311. constructor TJclPeImportLibItem.Create(AImage: TJclPeImage;
  1312. AImportDescriptor: Pointer; AImportKind: TJclPeImportKind; const AName: string;
  1313. AThunk: Pointer);
  1314. begin
  1315. inherited Create(AImage);
  1316. FTotalResolveCheck := icNotChecked;
  1317. FImportDescriptor := AImportDescriptor;
  1318. FImportKind := AImportKind;
  1319. FName := AName;
  1320. FThunk := AThunk;
  1321. FThunkData := AThunk;
  1322. end;
  1323. procedure TJclPeImportLibItem.CheckImports(ExportImage: TJclPeImage);
  1324. var
  1325. I: Integer;
  1326. ExportList: TJclPeExportFuncList;
  1327. begin
  1328. if ExportImage.StatusOK then
  1329. begin
  1330. FTotalResolveCheck := icResolved;
  1331. ExportList := ExportImage.ExportList;
  1332. for I := 0 to Count - 1 do
  1333. begin
  1334. with Items[I] do
  1335. if IsByOrdinal then
  1336. begin
  1337. if ExportList.OrdinalValid(Ordinal) then
  1338. SetResolveCheck(icResolved)
  1339. else
  1340. begin
  1341. SetResolveCheck(icUnresolved);
  1342. Self.FTotalResolveCheck := icUnresolved;
  1343. end;
  1344. end
  1345. else
  1346. begin
  1347. if ExportList.ItemFromName[Items[I].Name] <> nil then
  1348. SetResolveCheck(icResolved)
  1349. else
  1350. begin
  1351. SetResolveCheck(icUnresolved);
  1352. Self.FTotalResolveCheck := icUnresolved;
  1353. end;
  1354. end;
  1355. end;
  1356. end
  1357. else
  1358. begin
  1359. FTotalResolveCheck := icUnresolved;
  1360. for I := 0 to Count - 1 do
  1361. Items[I].SetResolveCheck(icUnresolved);
  1362. end;
  1363. end;
  1364. procedure TJclPeImportLibItem.CreateList;
  1365. procedure CreateList32;
  1366. var
  1367. Thunk32: PImageThunkData32;
  1368. OrdinalName: PImageImportByName;
  1369. Ordinal, Hint: Word;
  1370. Name: PAnsiChar;
  1371. ImportName: string;
  1372. begin
  1373. Thunk32 := PImageThunkData32(FThunk);
  1374. while Thunk32^.Function_ <> 0 do
  1375. begin
  1376. Ordinal := 0;
  1377. Hint := 0;
  1378. Name := nil;
  1379. if Thunk32^.Ordinal and IMAGE_ORDINAL_FLAG32 = 0 then
  1380. begin
  1381. case ImportKind of
  1382. ikImport, ikBoundImport:
  1383. begin
  1384. OrdinalName := PImageImportByName(Image.RvaToVa(Thunk32^.AddressOfData));
  1385. Hint := OrdinalName.Hint;
  1386. Name := OrdinalName.Name;
  1387. end;
  1388. ikDelayImport:
  1389. begin
  1390. OrdinalName := PImageImportByName(Image.RvaToVaEx(Thunk32^.AddressOfData));
  1391. Hint := OrdinalName.Hint;
  1392. Name := OrdinalName.Name;
  1393. end;
  1394. end;
  1395. end
  1396. else
  1397. Ordinal := IMAGE_ORDINAL32(Thunk32^.Ordinal);
  1398. if not TryUTF8ToString(Name, ImportName) then
  1399. ImportName := string(Name);
  1400. Add(TJclPeImportFuncItem.Create(Self, Ordinal, Hint, ImportName));
  1401. Inc(Thunk32);
  1402. end;
  1403. end;
  1404. procedure CreateList64;
  1405. var
  1406. Thunk64: PImageThunkData64;
  1407. OrdinalName: PImageImportByName;
  1408. Ordinal, Hint: Word;
  1409. Name: PAnsiChar;
  1410. ImportName: string;
  1411. begin
  1412. Thunk64 := PImageThunkData64(FThunk);
  1413. while Thunk64^.Function_ <> 0 do
  1414. begin
  1415. Ordinal := 0;
  1416. Hint := 0;
  1417. Name := nil;
  1418. if Thunk64^.Ordinal and IMAGE_ORDINAL_FLAG64 = 0 then
  1419. begin
  1420. case ImportKind of
  1421. ikImport, ikBoundImport:
  1422. begin
  1423. OrdinalName := PImageImportByName(Image.RvaToVa(Thunk64^.AddressOfData));
  1424. Hint := OrdinalName.Hint;
  1425. Name := OrdinalName.Name;
  1426. end;
  1427. ikDelayImport:
  1428. begin
  1429. OrdinalName := PImageImportByName(Image.RvaToVaEx(Thunk64^.AddressOfData));
  1430. Hint := OrdinalName.Hint;
  1431. Name := OrdinalName.Name;
  1432. end;
  1433. end;
  1434. end
  1435. else
  1436. Ordinal := IMAGE_ORDINAL64(Thunk64^.Ordinal);
  1437. if not TryUTF8ToString(Name, ImportName) then
  1438. ImportName := string(Name);
  1439. Add(TJclPeImportFuncItem.Create(Self, Ordinal, Hint, ImportName));
  1440. Inc(Thunk64);
  1441. end;
  1442. end;
  1443. begin
  1444. if FThunk = nil then
  1445. Exit;
  1446. case Image.Target of
  1447. taWin32:
  1448. CreateList32;
  1449. taWin64:
  1450. CreateList64;
  1451. end;
  1452. FThunk := nil;
  1453. end;
  1454. function TJclPeImportLibItem.GetCount: Integer;
  1455. begin
  1456. if FThunk <> nil then
  1457. CreateList;
  1458. Result := inherited Count;
  1459. end;
  1460. function TJclPeImportLibItem.GetFileName: TFileName;
  1461. begin
  1462. Result := Image.ExpandModuleName(Name);
  1463. end;
  1464. function TJclPeImportLibItem.GetItems(Index: Integer): TJclPeImportFuncItem;
  1465. begin
  1466. Result := TJclPeImportFuncItem(Get(Index));
  1467. end;
  1468. function TJclPeImportLibItem.GetName: string;
  1469. begin
  1470. Result := AnsiLowerCase(OriginalName);
  1471. end;
  1472. function TJclPeImportLibItem.GetThunkData32: PImageThunkData32;
  1473. begin
  1474. if Image.Target = taWin32 then
  1475. Result := FThunkData
  1476. else
  1477. Result := nil;
  1478. end;
  1479. function TJclPeImportLibItem.GetThunkData64: PImageThunkData64;
  1480. begin
  1481. if Image.Target = taWin64 then
  1482. Result := FThunkData
  1483. else
  1484. Result := nil;
  1485. end;
  1486. procedure TJclPeImportLibItem.SetImportDirectoryIndex(Value: Integer);
  1487. begin
  1488. FImportDirectoryIndex := Value;
  1489. end;
  1490. procedure TJclPeImportLibItem.SetImportKind(Value: TJclPeImportKind);
  1491. begin
  1492. FImportKind := Value;
  1493. end;
  1494. procedure TJclPeImportLibItem.SetSorted(Value: Boolean);
  1495. begin
  1496. FSorted := Value;
  1497. end;
  1498. procedure TJclPeImportLibItem.SetThunk(Value: Pointer);
  1499. begin
  1500. FThunk := Value;
  1501. FThunkData := Value;
  1502. end;
  1503. procedure TJclPeImportLibItem.SortList(SortType: TJclPeImportSort; Descending: Boolean);
  1504. begin
  1505. if not FSorted or (SortType <> FLastSortType) or (Descending <> FLastSortDescending) then
  1506. begin
  1507. GetCount; // create list if it wasn't created
  1508. Sort(GetImportSortFunction(SortType, Descending));
  1509. FLastSortType := SortType;
  1510. FLastSortDescending := Descending;
  1511. FSorted := True;
  1512. end;
  1513. end;
  1514. //=== { TJclPeImportList } ===================================================
  1515. constructor TJclPeImportList.Create(AImage: TJclPeImage);
  1516. begin
  1517. inherited Create(AImage);
  1518. FAllItemsList := TList.Create;
  1519. FAllItemsList.Capacity := 256;
  1520. FUniqueNamesList := TStringList.Create;
  1521. FUniqueNamesList.Sorted := True;
  1522. FUniqueNamesList.Duplicates := dupIgnore;
  1523. FLastAllSortType := isName;
  1524. FLastAllSortDescending := False;
  1525. CreateList;
  1526. end;
  1527. destructor TJclPeImportList.Destroy;
  1528. var
  1529. I: Integer;
  1530. begin
  1531. FreeAndNil(FAllItemsList);
  1532. FreeAndNil(FUniqueNamesList);
  1533. for I := 0 to Length(FparallelImportTable) - 1 do
  1534. FreeMem(FparallelImportTable[I]);
  1535. inherited Destroy;
  1536. end;
  1537. procedure TJclPeImportList.CheckImports(PeImageCache: TJclPeImagesCache);
  1538. var
  1539. I: Integer;
  1540. ExportPeImage: TJclPeImage;
  1541. begin
  1542. Image.CheckNotAttached;
  1543. if PeImageCache <> nil then
  1544. ExportPeImage := nil // to make the compiler happy
  1545. else
  1546. ExportPeImage := TJclPeImage.Create(True);
  1547. try
  1548. for I := 0 to Count - 1 do
  1549. if Items[I].TotalResolveCheck = icNotChecked then
  1550. begin
  1551. if PeImageCache <> nil then
  1552. ExportPeImage := PeImageCache[Items[I].FileName]
  1553. else
  1554. ExportPeImage.FileName := Items[I].FileName;
  1555. ExportPeImage.ExportList.PrepareForFastNameSearch;
  1556. Items[I].CheckImports(ExportPeImage);
  1557. end;
  1558. finally
  1559. if PeImageCache = nil then
  1560. ExportPeImage.Free;
  1561. end;
  1562. end;
  1563. procedure TJclPeImportList.CreateList;
  1564. procedure CreateDelayImportList32(DelayImportDesc: PImgDelayDescrV1);
  1565. var
  1566. LibItem: TJclPeImportLibItem;
  1567. UTF8Name: TUTF8String;
  1568. LibName: string;
  1569. begin
  1570. while DelayImportDesc^.szName <> nil do
  1571. begin
  1572. UTF8Name := PAnsiChar(Image.RvaToVaEx(DWORD(DelayImportDesc^.szName)));
  1573. if not TryUTF8ToString(UTF8Name, LibName) then
  1574. LibName := string(UTF8Name);
  1575. LibItem := TJclPeImportLibItem.Create(Image, DelayImportDesc, ikDelayImport,
  1576. LibName, Image.RvaToVaEx(DWORD(DelayImportDesc^.pINT)));
  1577. Add(LibItem);
  1578. FUniqueNamesList.AddObject(AnsiLowerCase(LibItem.Name), LibItem);
  1579. Inc(DelayImportDesc);
  1580. end;
  1581. end;
  1582. procedure CreateDelayImportList64(DelayImportDesc: PImgDelayDescrV2);
  1583. var
  1584. LibItem: TJclPeImportLibItem;
  1585. UTF8Name: TUTF8String;
  1586. LibName: string;
  1587. begin
  1588. while DelayImportDesc^.rvaDLLName <> 0 do
  1589. begin
  1590. UTF8Name := PAnsiChar(Image.RvaToVa(DelayImportDesc^.rvaDLLName));
  1591. if not TryUTF8ToString(UTF8Name, LibName) then
  1592. LibName := string(UTF8Name);
  1593. LibItem := TJclPeImportLibItem.Create(Image, DelayImportDesc, ikDelayImport,
  1594. LibName, Image.RvaToVa(DelayImportDesc^.rvaINT));
  1595. Add(LibItem);
  1596. FUniqueNamesList.AddObject(AnsiLowerCase(LibItem.Name), LibItem);
  1597. Inc(DelayImportDesc);
  1598. end;
  1599. end;
  1600. var
  1601. ImportDesc: PImageImportDescriptor;
  1602. LibItem: TJclPeImportLibItem;
  1603. UTF8Name: TUTF8String;
  1604. LibName, ModuleName: string;
  1605. DelayImportDesc: Pointer;
  1606. BoundImports, BoundImport: PImageBoundImportDescriptor;
  1607. S: string;
  1608. I: Integer;
  1609. Thunk: Pointer;
  1610. begin
  1611. SetCapacity(100);
  1612. with Image do
  1613. begin
  1614. if not StatusOK then
  1615. Exit;
  1616. ImportDesc := DirectoryEntryToData(IMAGE_DIRECTORY_ENTRY_IMPORT);
  1617. if ImportDesc <> nil then
  1618. while ImportDesc^.Name <> 0 do
  1619. begin
  1620. if ImportDesc^.Union.Characteristics = 0 then
  1621. begin
  1622. if AttachedImage then // Borland images doesn't have two parallel arrays
  1623. Thunk := nil // see MakeBorlandImportTableForMappedImage method
  1624. else
  1625. Thunk := RvaToVa(ImportDesc^.FirstThunk);
  1626. FLinkerProducer := lrBorland;
  1627. end
  1628. else
  1629. begin
  1630. Thunk := RvaToVa(ImportDesc^.Union.Characteristics);
  1631. FLinkerProducer := lrMicrosoft;
  1632. end;
  1633. UTF8Name := PAnsiChar(RvaToVa(ImportDesc^.Name));
  1634. if not TryUTF8ToString(UTF8Name, LibName) then
  1635. LibName := string(UTF8Name);
  1636. LibItem := TJclPeImportLibItem.Create(Image, ImportDesc, ikImport, LibName, Thunk);
  1637. Add(LibItem);
  1638. FUniqueNamesList.AddObject(AnsiLowerCase(LibItem.Name), LibItem);
  1639. Inc(ImportDesc);
  1640. end;
  1641. DelayImportDesc := DirectoryEntryToData(IMAGE_DIRECTORY_ENTRY_DELAY_IMPORT);
  1642. if DelayImportDesc <> nil then
  1643. begin
  1644. case Target of
  1645. taWin32:
  1646. CreateDelayImportList32(DelayImportDesc);
  1647. taWin64:
  1648. CreateDelayImportList64(DelayImportDesc);
  1649. end;
  1650. end;
  1651. BoundImports := DirectoryEntryToData(IMAGE_DIRECTORY_ENTRY_BOUND_IMPORT);
  1652. if BoundImports <> nil then
  1653. begin
  1654. BoundImport := BoundImports;
  1655. while BoundImport^.OffsetModuleName <> 0 do
  1656. begin
  1657. UTF8Name := PAnsiChar(TJclAddr(BoundImports) + BoundImport^.OffsetModuleName);
  1658. if not TryUTF8ToString(UTF8Name, ModuleName) then
  1659. ModuleName := string(UTF8Name);
  1660. S := AnsiLowerCase(ModuleName);
  1661. I := FUniqueNamesList.IndexOf(S);
  1662. if I >= 0 then
  1663. TJclPeImportLibItem(FUniqueNamesList.Objects[I]).SetImportKind(ikBoundImport);
  1664. for I := 1 to BoundImport^.NumberOfModuleForwarderRefs do
  1665. Inc(PImageBoundForwarderRef(BoundImport)); // skip forward information
  1666. Inc(BoundImport);
  1667. end;
  1668. end;
  1669. end;
  1670. for I := 0 to Count - 1 do
  1671. Items[I].SetImportDirectoryIndex(I);
  1672. end;
  1673. function TJclPeImportList.GetAllItemCount: Integer;
  1674. begin
  1675. Result := FAllItemsList.Count;
  1676. if Result = 0 then // we haven't created the list yet -> create unsorted list
  1677. begin
  1678. RefreshAllItems;
  1679. Result := FAllItemsList.Count;
  1680. end;
  1681. end;
  1682. function TJclPeImportList.GetAllItems(Index: Integer): TJclPeImportFuncItem;
  1683. begin
  1684. Result := TJclPeImportFuncItem(FAllItemsList[Index]);
  1685. end;
  1686. function TJclPeImportList.GetItems(Index: Integer): TJclPeImportLibItem;
  1687. begin
  1688. Result := TJclPeImportLibItem(Get(Index));
  1689. end;
  1690. function TJclPeImportList.GetUniqueLibItemCount: Integer;
  1691. begin
  1692. Result := FUniqueNamesList.Count;
  1693. end;
  1694. function TJclPeImportList.GetUniqueLibItemFromName(const Name: string): TJclPeImportLibItem;
  1695. var
  1696. I: Integer;
  1697. begin
  1698. I := FUniqueNamesList.IndexOf(Name);
  1699. if I = -1 then
  1700. Result := nil
  1701. else
  1702. Result := TJclPeImportLibItem(FUniqueNamesList.Objects[I]);
  1703. end;
  1704. function TJclPeImportList.GetUniqueLibItems(Index: Integer): TJclPeImportLibItem;
  1705. begin
  1706. Result := TJclPeImportLibItem(FUniqueNamesList.Objects[Index]);
  1707. end;
  1708. function TJclPeImportList.GetUniqueLibNames(Index: Integer): string;
  1709. begin
  1710. Result := FUniqueNamesList[Index];
  1711. end;
  1712. function TJclPeImportList.MakeBorlandImportTableForMappedImage: Boolean;
  1713. var
  1714. FileImage: TJclPeImage;
  1715. I, TableSize: Integer;
  1716. begin
  1717. if Image.AttachedImage and (LinkerProducer = lrBorland) and
  1718. (Length(FParallelImportTable) = 0) then
  1719. begin
  1720. FileImage := TJclPeImage.Create(True);
  1721. try
  1722. FileImage.FileName := Image.FileName;
  1723. Result := FileImage.StatusOK;
  1724. if Result then
  1725. begin
  1726. SetLength(FParallelImportTable, FileImage.ImportList.Count);
  1727. for I := 0 to FileImage.ImportList.Count - 1 do
  1728. begin
  1729. Assert(Items[I].ImportKind = ikImport); // Borland doesn't have Delay load or Bound imports
  1730. TableSize := (FileImage.ImportList[I].Count + 1);
  1731. case Image.Target of
  1732. taWin32:
  1733. begin
  1734. TableSize := TableSize * SizeOf(TImageThunkData32);
  1735. GetMem(FParallelImportTable[I], TableSize);
  1736. System.Move(FileImage.ImportList[I].ThunkData32^, FParallelImportTable[I]^, TableSize);
  1737. Items[I].SetThunk(FParallelImportTable[I]);
  1738. end;
  1739. taWin64:
  1740. begin
  1741. TableSize := TableSize * SizeOf(TImageThunkData64);
  1742. GetMem(FParallelImportTable[I], TableSize);
  1743. System.Move(FileImage.ImportList[I].ThunkData64^, FParallelImportTable[I]^, TableSize);
  1744. Items[I].SetThunk(FParallelImportTable[I]);
  1745. end;
  1746. end;
  1747. end;
  1748. end;
  1749. finally
  1750. FileImage.Free;
  1751. end;
  1752. end
  1753. else
  1754. Result := True;
  1755. end;
  1756. procedure TJclPeImportList.RefreshAllItems;
  1757. var
  1758. L, I: Integer;
  1759. LibItem: TJclPeImportLibItem;
  1760. begin
  1761. FAllItemsList.Clear;
  1762. for L := 0 to Count - 1 do
  1763. begin
  1764. LibItem := Items[L];
  1765. if (Length(FFilterModuleName) = 0) or (AnsiCompareText(LibItem.Name, FFilterModuleName) = 0) then
  1766. for I := 0 to LibItem.Count - 1 do
  1767. FAllItemsList.Add(LibItem[I]);
  1768. end;
  1769. end;
  1770. procedure TJclPeImportList.SetFilterModuleName(const Value: string);
  1771. begin
  1772. if (FFilterModuleName <> Value) or (FAllItemsList.Count = 0) then
  1773. begin
  1774. FFilterModuleName := Value;
  1775. RefreshAllItems;
  1776. FAllItemsList.Sort(GetImportSortFunction(FLastAllSortType, FLastAllSortDescending));
  1777. end;
  1778. end;
  1779. function TJclPeImportList.SmartFindName(const CompareName, LibName: string;
  1780. Options: TJclSmartCompOptions): TJclPeImportFuncItem;
  1781. var
  1782. L, I: Integer;
  1783. LibItem: TJclPeImportLibItem;
  1784. begin
  1785. Result := nil;
  1786. for L := 0 to Count - 1 do
  1787. begin
  1788. LibItem := Items[L];
  1789. if (Length(LibName) = 0) or (AnsiCompareText(LibItem.Name, LibName) = 0) then
  1790. for I := 0 to LibItem.Count - 1 do
  1791. if PeSmartFunctionNameSame(CompareName, LibItem[I].Name, Options) then
  1792. begin
  1793. Result := LibItem[I];
  1794. Break;
  1795. end;
  1796. end;
  1797. end;
  1798. procedure TJclPeImportList.SortAllItemsList(SortType: TJclPeImportSort; Descending: Boolean);
  1799. begin
  1800. GetAllItemCount; // create list if it wasn't created
  1801. FAllItemsList.Sort(GetImportSortFunction(SortType, Descending));
  1802. FLastAllSortType := SortType;
  1803. FLastAllSortDescending := Descending;
  1804. end;
  1805. procedure TJclPeImportList.SortList(SortType: TJclPeImportLibSort);
  1806. begin
  1807. Sort(GetImportLibSortFunction(SortType));
  1808. end;
  1809. procedure TJclPeImportList.TryGetNamesForOrdinalImports;
  1810. var
  1811. LibNamesList: TStringList;
  1812. L, I: Integer;
  1813. LibPeDump: TJclPeImage;
  1814. procedure TryGetNames(const ModuleName: string);
  1815. var
  1816. Item: TJclPeImportFuncItem;
  1817. I, L: Integer;
  1818. ImportLibItem: TJclPeImportLibItem;
  1819. ExportItem: TJclPeExportFuncItem;
  1820. ExportList: TJclPeExportFuncList;
  1821. begin
  1822. if Image.AttachedImage then
  1823. LibPeDump.AttachLoadedModule(GetModuleHandle(PChar(ModuleName)))
  1824. else
  1825. LibPeDump.FileName := Image.ExpandModuleName(ModuleName);
  1826. if not LibPeDump.StatusOK then
  1827. Exit;
  1828. ExportList := LibPeDump.ExportList;
  1829. for L := 0 to Count - 1 do
  1830. begin
  1831. ImportLibItem := Items[L];
  1832. if AnsiCompareText(ImportLibItem.Name, ModuleName) = 0 then
  1833. begin
  1834. for I := 0 to ImportLibItem.Count - 1 do
  1835. begin
  1836. Item := ImportLibItem[I];
  1837. if Item.IsByOrdinal then
  1838. begin
  1839. ExportItem := ExportList.ItemFromOrdinal[Item.Ordinal];
  1840. if (ExportItem <> nil) and (ExportItem.Name <> '') then
  1841. Item.SetIndirectImportName(ExportItem.Name);
  1842. end;
  1843. end;
  1844. ImportLibItem.SetSorted(False);
  1845. end;
  1846. end;
  1847. end;
  1848. begin
  1849. LibNamesList := TStringList.Create;
  1850. try
  1851. LibNamesList.Sorted := True;
  1852. LibNamesList.Duplicates := dupIgnore;
  1853. for L := 0 to Count - 1 do
  1854. with Items[L] do
  1855. for I := 0 to Count - 1 do
  1856. if Items[I].IsByOrdinal then
  1857. LibNamesList.Add(AnsiUpperCase(Name));
  1858. LibPeDump := TJclPeImage.Create(True);
  1859. try
  1860. for I := 0 to LibNamesList.Count - 1 do
  1861. TryGetNames(LibNamesList[I]);
  1862. finally
  1863. LibPeDump.Free;
  1864. end;
  1865. SortAllItemsList(FLastAllSortType, FLastAllSortDescending);
  1866. finally
  1867. LibNamesList.Free;
  1868. end;
  1869. end;
  1870. //=== { TJclPeExportFuncItem } ===============================================
  1871. constructor TJclPeExportFuncItem.Create(AExportList: TJclPeExportFuncList;
  1872. const AName, AForwardedName: string; AAddress: DWORD; AHint: Word;
  1873. AOrdinal: Word; AResolveCheck: TJclPeResolveCheck);
  1874. var
  1875. DotPos: Integer;
  1876. begin
  1877. inherited Create;
  1878. FExportList := AExportList;
  1879. FName := AName;
  1880. FForwardedName := AForwardedName;
  1881. FAddress := AAddress;
  1882. FHint := AHint;
  1883. FOrdinal := AOrdinal;
  1884. FResolveCheck := AResolveCheck;
  1885. DotPos := AnsiPos('.', ForwardedName);
  1886. if DotPos > 0 then
  1887. FForwardedDotPos := Copy(ForwardedName, DotPos + 1, Length(ForwardedName) - DotPos)
  1888. else
  1889. FForwardedDotPos := '';
  1890. end;
  1891. function TJclPeExportFuncItem.GetAddressOrForwardStr: string;
  1892. begin
  1893. if IsForwarded then
  1894. Result := ForwardedName
  1895. else
  1896. FmtStr(Result, '%.8x', [Address]);
  1897. end;
  1898. function TJclPeExportFuncItem.GetForwardedFuncName: string;
  1899. begin
  1900. if (Length(FForwardedDotPos) > 0) and (FForwardedDotPos[1] <> '#') then
  1901. Result := FForwardedDotPos
  1902. else
  1903. Result := '';
  1904. end;
  1905. function TJclPeExportFuncItem.GetForwardedFuncOrdinal: DWORD;
  1906. begin
  1907. if (Length(FForwardedDotPos) > 0) and (FForwardedDotPos[1] = '#') then
  1908. Result := StrToIntDef(FForwardedDotPos, 0)
  1909. else
  1910. Result := 0;
  1911. end;
  1912. function TJclPeExportFuncItem.GetForwardedLibName: string;
  1913. begin
  1914. if Length(FForwardedDotPos) = 0 then
  1915. Result := ''
  1916. else
  1917. Result := AnsiLowerCase(Copy(FForwardedName, 1, Length(FForwardedName) - Length(FForwardedDotPos) - 1)) + BinaryExtensionLibrary;
  1918. end;
  1919. function TJclPeExportFuncItem.GetIsExportedVariable: Boolean;
  1920. begin
  1921. case FExportList.Image.Target of
  1922. taWin32:
  1923. begin
  1924. {$IFDEF DELPHI64_TEMPORARY}
  1925. System.Error(rePlatformNotImplemented);//there is no BaseOfData in the 32-bit header for Win64
  1926. Result := False;
  1927. {$ELSE ~DELPHI64_TEMPORARY}
  1928. Result := (Address >= FExportList.Image.OptionalHeader32.BaseOfData);
  1929. {$ENDIF ~DELPHI64_TEMPORARY}
  1930. end;
  1931. taWin64:
  1932. Result := False;
  1933. // TODO equivalent for 64-bit modules
  1934. //Result := (Address >= FExportList.Image.OptionalHeader64.BaseOfData);
  1935. else
  1936. Result := False;
  1937. end;
  1938. end;
  1939. function TJclPeExportFuncItem.GetIsForwarded: Boolean;
  1940. begin
  1941. Result := Length(FForwardedName) <> 0;
  1942. end;
  1943. function TJclPeExportFuncItem.GetMappedAddress: Pointer;
  1944. begin
  1945. Result := FExportList.Image.RvaToVa(FAddress);
  1946. end;
  1947. function TJclPeExportFuncItem.GetSectionName: string;
  1948. begin
  1949. if IsForwarded then
  1950. Result := ''
  1951. else
  1952. with FExportList.Image do
  1953. Result := ImageSectionNameFromRva[Address];
  1954. end;
  1955. procedure TJclPeExportFuncItem.SetResolveCheck(Value: TJclPeResolveCheck);
  1956. begin
  1957. FResolveCheck := Value;
  1958. end;
  1959. // Export sort functions
  1960. function ExportSortByName(Item1, Item2: Pointer): Integer;
  1961. begin
  1962. Result := CompareStr(TJclPeExportFuncItem(Item1).Name, TJclPeExportFuncItem(Item2).Name);
  1963. end;
  1964. function ExportSortByNameDESC(Item1, Item2: Pointer): Integer;
  1965. begin
  1966. Result := ExportSortByName(Item2, Item1);
  1967. end;
  1968. function ExportSortByOrdinal(Item1, Item2: Pointer): Integer;
  1969. begin
  1970. Result := TJclPeExportFuncItem(Item1).Ordinal - TJclPeExportFuncItem(Item2).Ordinal;
  1971. end;
  1972. function ExportSortByOrdinalDESC(Item1, Item2: Pointer): Integer;
  1973. begin
  1974. Result := ExportSortByOrdinal(Item2, Item1);
  1975. end;
  1976. function ExportSortByHint(Item1, Item2: Pointer): Integer;
  1977. begin
  1978. Result := TJclPeExportFuncItem(Item1).Hint - TJclPeExportFuncItem(Item2).Hint;
  1979. end;
  1980. function ExportSortByHintDESC(Item1, Item2: Pointer): Integer;
  1981. begin
  1982. Result := ExportSortByHint(Item2, Item1);
  1983. end;
  1984. function ExportSortByAddress(Item1, Item2: Pointer): Integer;
  1985. begin
  1986. Result := INT_PTR(TJclPeExportFuncItem(Item1).Address) - INT_PTR(TJclPeExportFuncItem(Item2).Address);
  1987. if Result = 0 then
  1988. Result := ExportSortByName(Item1, Item2);
  1989. end;
  1990. function ExportSortByAddressDESC(Item1, Item2: Pointer): Integer;
  1991. begin
  1992. Result := ExportSortByAddress(Item2, Item1);
  1993. end;
  1994. function ExportSortByForwarded(Item1, Item2: Pointer): Integer;
  1995. begin
  1996. Result := CompareStr(TJclPeExportFuncItem(Item1).ForwardedName, TJclPeExportFuncItem(Item2).ForwardedName);
  1997. if Result = 0 then
  1998. Result := ExportSortByName(Item1, Item2);
  1999. end;
  2000. function ExportSortByForwardedDESC(Item1, Item2: Pointer): Integer;
  2001. begin
  2002. Result := ExportSortByForwarded(Item2, Item1);
  2003. end;
  2004. function ExportSortByAddrOrFwd(Item1, Item2: Pointer): Integer;
  2005. begin
  2006. Result := CompareStr(TJclPeExportFuncItem(Item1).AddressOrForwardStr, TJclPeExportFuncItem(Item2).AddressOrForwardStr);
  2007. end;
  2008. function ExportSortByAddrOrFwdDESC(Item1, Item2: Pointer): Integer;
  2009. begin
  2010. Result := ExportSortByAddrOrFwd(Item2, Item1);
  2011. end;
  2012. function ExportSortBySection(Item1, Item2: Pointer): Integer;
  2013. begin
  2014. Result := CompareStr(TJclPeExportFuncItem(Item1).SectionName, TJclPeExportFuncItem(Item2).SectionName);
  2015. if Result = 0 then
  2016. Result := ExportSortByName(Item1, Item2);
  2017. end;
  2018. function ExportSortBySectionDESC(Item1, Item2: Pointer): Integer;
  2019. begin
  2020. Result := ExportSortBySection(Item2, Item1);
  2021. end;
  2022. //=== { TJclPeExportFuncList } ===============================================
  2023. constructor TJclPeExportFuncList.Create(AImage: TJclPeImage);
  2024. begin
  2025. inherited Create(AImage);
  2026. FTotalResolveCheck := icNotChecked;
  2027. CreateList;
  2028. end;
  2029. destructor TJclPeExportFuncList.Destroy;
  2030. begin
  2031. FreeAndNil(FForwardedLibsList);
  2032. inherited Destroy;
  2033. end;
  2034. function TJclPeExportFuncList.CanPerformFastNameSearch: Boolean;
  2035. begin
  2036. Result := FSorted and (FLastSortType = esName) and not FLastSortDescending;
  2037. end;
  2038. procedure TJclPeExportFuncList.CheckForwards(PeImageCache: TJclPeImagesCache);
  2039. var
  2040. I: Integer;
  2041. FullFileName: TFileName;
  2042. ForwardPeImage: TJclPeImage;
  2043. ModuleResolveCheck: TJclPeResolveCheck;
  2044. procedure PerformCheck(const ModuleName: string);
  2045. var
  2046. I: Integer;
  2047. Item: TJclPeExportFuncItem;
  2048. EL: TJclPeExportFuncList;
  2049. begin
  2050. EL := ForwardPeImage.ExportList;
  2051. EL.PrepareForFastNameSearch;
  2052. ModuleResolveCheck := icResolved;
  2053. for I := 0 to Count - 1 do
  2054. begin
  2055. Item := Items[I];
  2056. if (not Item.IsForwarded) or (Item.ResolveCheck <> icNotChecked) or
  2057. (Item.ForwardedLibName <> ModuleName) then
  2058. Continue;
  2059. if EL.ItemFromName[Item.ForwardedFuncName] = nil then
  2060. begin
  2061. Item.SetResolveCheck(icUnresolved);
  2062. ModuleResolveCheck := icUnresolved;
  2063. end
  2064. else
  2065. Item.SetResolveCheck(icResolved);
  2066. end;
  2067. end;
  2068. begin
  2069. if not AnyForwards then
  2070. Exit;
  2071. FTotalResolveCheck := icResolved;
  2072. if PeImageCache <> nil then
  2073. ForwardPeImage := nil // to make the compiler happy
  2074. else
  2075. ForwardPeImage := TJclPeImage.Create(True);
  2076. try
  2077. for I := 0 to ForwardedLibsList.Count - 1 do
  2078. begin
  2079. FullFileName := Image.ExpandModuleName(ForwardedLibsList[I]);
  2080. if PeImageCache <> nil then
  2081. ForwardPeImage := PeImageCache[FullFileName]
  2082. else
  2083. ForwardPeImage.FileName := FullFileName;
  2084. if ForwardPeImage.StatusOK then
  2085. PerformCheck(ForwardedLibsList[I])
  2086. else
  2087. ModuleResolveCheck := icUnresolved;
  2088. FForwardedLibsList.Objects[I] := Pointer(ModuleResolveCheck);
  2089. if ModuleResolveCheck = icUnresolved then
  2090. FTotalResolveCheck := icUnresolved;
  2091. end;
  2092. finally
  2093. if PeImageCache = nil then
  2094. ForwardPeImage.Free;
  2095. end;
  2096. end;
  2097. procedure TJclPeExportFuncList.CreateList;
  2098. var
  2099. Functions: Pointer;
  2100. Address, NameCount: DWORD;
  2101. NameOrdinals: PWORD;
  2102. Names: PDWORD;
  2103. I: Integer;
  2104. ExportItem: TJclPeExportFuncItem;
  2105. ExportVABegin, ExportVAEnd: DWORD;
  2106. UTF8Name: TUTF8String;
  2107. ForwardedName, ExportName: string;
  2108. begin
  2109. with Image do
  2110. begin
  2111. if not StatusOK then
  2112. Exit;
  2113. with Directories[IMAGE_DIRECTORY_ENTRY_EXPORT] do
  2114. begin
  2115. ExportVABegin := VirtualAddress;
  2116. ExportVAEnd := VirtualAddress + TJclAddr(Size);
  2117. end;
  2118. FExportDir := DirectoryEntryToData(IMAGE_DIRECTORY_ENTRY_EXPORT);
  2119. if FExportDir <> nil then
  2120. begin
  2121. FBase := FExportDir^.Base;
  2122. FFunctionCount := FExportDir^.NumberOfFunctions;
  2123. Functions := RvaToVa(FExportDir^.AddressOfFunctions);
  2124. NameOrdinals := RvaToVa(FExportDir^.AddressOfNameOrdinals);
  2125. Names := RvaToVa(FExportDir^.AddressOfNames);
  2126. NameCount := FExportDir^.NumberOfNames;
  2127. Count := FExportDir^.NumberOfFunctions;
  2128. for I := 0 to Count - 1 do
  2129. begin
  2130. Address := PDWORD(TJclAddr(Functions) + TJclAddr(I) * SizeOf(DWORD))^;
  2131. if (Address >= ExportVABegin) and (Address <= ExportVAEnd) then
  2132. begin
  2133. FAnyForwards := True;
  2134. UTF8Name := PAnsiChar(RvaToVa(Address));
  2135. if not TryUTF8ToString(UTF8Name, ForwardedName) then
  2136. ForwardedName := string(UTF8Name);
  2137. end
  2138. else
  2139. ForwardedName := '';
  2140. ExportItem := TJclPeExportFuncItem.Create(Self, '',
  2141. ForwardedName, Address, $FFFF, TJclAddr(I) + FBase, icNotChecked);
  2142. List{$IFNDEF RTL230_UP}^{$ENDIF !RTL230_UP}[I] := ExportItem;
  2143. end;
  2144. for I := 0 to NameCount - 1 do
  2145. begin
  2146. // named function
  2147. UTF8Name := PAnsiChar(RvaToVa(Names^));
  2148. if not TryUTF8ToString(UTF8Name, ExportName) then
  2149. ExportName := string(UTF8Name);
  2150. ExportItem := TJclPeExportFuncItem(List{$IFNDEF RTL230_UP}^{$ENDIF !RTL230_UP}[NameOrdinals^]);
  2151. ExportItem.FName := ExportName;
  2152. ExportItem.FHint := I;
  2153. Inc(NameOrdinals);
  2154. Inc(Names);
  2155. end;
  2156. end;
  2157. end;
  2158. end;
  2159. function TJclPeExportFuncList.GetForwardedLibsList: TStrings;
  2160. var
  2161. I: Integer;
  2162. begin
  2163. if FForwardedLibsList = nil then
  2164. begin
  2165. FForwardedLibsList := TStringList.Create;
  2166. FForwardedLibsList.Sorted := True;
  2167. FForwardedLibsList.Duplicates := dupIgnore;
  2168. if FAnyForwards then
  2169. for I := 0 to Count - 1 do
  2170. with Items[I] do
  2171. if IsForwarded then
  2172. FForwardedLibsList.AddObject(ForwardedLibName, Pointer(icNotChecked));
  2173. end;
  2174. Result := FForwardedLibsList;
  2175. end;
  2176. function TJclPeExportFuncList.GetItemFromAddress(Address: DWORD): TJclPeExportFuncItem;
  2177. var
  2178. I: Integer;
  2179. begin
  2180. Result := nil;
  2181. for I := 0 to Count - 1 do
  2182. if Items[I].Address = Address then
  2183. begin
  2184. Result := Items[I];
  2185. Break;
  2186. end;
  2187. end;
  2188. function TJclPeExportFuncList.GetItemFromName(const Name: string): TJclPeExportFuncItem;
  2189. var
  2190. L, H, I, C: Integer;
  2191. B: Boolean;
  2192. begin
  2193. Result := nil;
  2194. if CanPerformFastNameSearch then
  2195. begin
  2196. L := 0;
  2197. H := Count - 1;
  2198. B := False;
  2199. while L <= H do
  2200. begin
  2201. I := (L + H) shr 1;
  2202. C := CompareStr(Items[I].Name, Name);
  2203. if C < 0 then
  2204. L := I + 1
  2205. else
  2206. begin
  2207. H := I - 1;
  2208. if C = 0 then
  2209. begin
  2210. B := True;
  2211. L := I;
  2212. end;
  2213. end;
  2214. end;
  2215. if B then
  2216. Result := Items[L];
  2217. end
  2218. else
  2219. for I := 0 to Count - 1 do
  2220. if Items[I].Name = Name then
  2221. begin
  2222. Result := Items[I];
  2223. Break;
  2224. end;
  2225. end;
  2226. function TJclPeExportFuncList.GetItemFromOrdinal(Ordinal: DWORD): TJclPeExportFuncItem;
  2227. var
  2228. I: Integer;
  2229. begin
  2230. Result := nil;
  2231. for I := 0 to Count - 1 do
  2232. if Items[I].Ordinal = Ordinal then
  2233. begin
  2234. Result := Items[I];
  2235. Break;
  2236. end;
  2237. end;
  2238. function TJclPeExportFuncList.GetItems(Index: Integer): TJclPeExportFuncItem;
  2239. begin
  2240. Result := TJclPeExportFuncItem(Get(Index));
  2241. end;
  2242. function TJclPeExportFuncList.GetName: string;
  2243. var
  2244. UTF8ExportName: TUTF8String;
  2245. begin
  2246. if (FExportDir = nil) or (FExportDir^.Name = 0) then
  2247. Result := ''
  2248. else
  2249. begin
  2250. UTF8ExportName := PAnsiChar(Image.RvaToVa(FExportDir^.Name));
  2251. if not TryUTF8ToString(UTF8ExportName, Result) then
  2252. Result := string(UTF8ExportName);
  2253. end;
  2254. end;
  2255. class function TJclPeExportFuncList.ItemName(Item: TJclPeExportFuncItem): string;
  2256. begin
  2257. if Item = nil then
  2258. Result := ''
  2259. else
  2260. Result := Item.Name;
  2261. end;
  2262. function TJclPeExportFuncList.OrdinalValid(Ordinal: DWORD): Boolean;
  2263. begin
  2264. Result := (FExportDir <> nil) and (Ordinal >= Base) and
  2265. (Ordinal < FunctionCount + Base);
  2266. end;
  2267. procedure TJclPeExportFuncList.PrepareForFastNameSearch;
  2268. begin
  2269. if not CanPerformFastNameSearch then
  2270. SortList(esName, False);
  2271. end;
  2272. function TJclPeExportFuncList.SmartFindName(const CompareName: string;
  2273. Options: TJclSmartCompOptions): TJclPeExportFuncItem;
  2274. var
  2275. I: Integer;
  2276. begin
  2277. Result := nil;
  2278. for I := 0 to Count - 1 do
  2279. begin
  2280. if PeSmartFunctionNameSame(CompareName, Items[I].Name, Options) then
  2281. begin
  2282. Result := Items[I];
  2283. Break;
  2284. end;
  2285. end;
  2286. end;
  2287. procedure TJclPeExportFuncList.SortList(SortType: TJclPeExportSort; Descending: Boolean);
  2288. const
  2289. SortFunctions: array [TJclPeExportSort, Boolean] of TListSortCompare =
  2290. ((ExportSortByName, ExportSortByNameDESC),
  2291. (ExportSortByOrdinal, ExportSortByOrdinalDESC),
  2292. (ExportSortByHint, ExportSortByHintDESC),
  2293. (ExportSortByAddress, ExportSortByAddressDESC),
  2294. (ExportSortByForwarded, ExportSortByForwardedDESC),
  2295. (ExportSortByAddrOrFwd, ExportSortByAddrOrFwdDESC),
  2296. (ExportSortBySection, ExportSortBySectionDESC)
  2297. );
  2298. begin
  2299. if not FSorted or (SortType <> FLastSortType) or (Descending <> FLastSortDescending) then
  2300. begin
  2301. Sort(SortFunctions[SortType, Descending]);
  2302. FLastSortType := SortType;
  2303. FLastSortDescending := Descending;
  2304. FSorted := True;
  2305. end;
  2306. end;
  2307. //=== { TJclPeResourceRawStream } ============================================
  2308. constructor TJclPeResourceRawStream.Create(AResourceItem: TJclPeResourceItem);
  2309. begin
  2310. Assert(not AResourceItem.IsDirectory);
  2311. inherited Create;
  2312. SetPointer(AResourceItem.RawEntryData, AResourceItem.RawEntryDataSize);
  2313. end;
  2314. function TJclPeResourceRawStream.Write(const Buffer; Count: Integer): Longint;
  2315. begin
  2316. raise EJclPeImageError.CreateRes(@RsPeReadOnlyStream);
  2317. end;
  2318. //=== { TJclPeResourceItem } =================================================
  2319. constructor TJclPeResourceItem.Create(AImage: TJclPeImage;
  2320. AParentItem: TJclPeResourceItem; AEntry: PImageResourceDirectoryEntry);
  2321. begin
  2322. inherited Create;
  2323. FImage := AImage;
  2324. FEntry := AEntry;
  2325. FParentItem := AParentItem;
  2326. if AParentItem = nil then
  2327. FLevel := 1
  2328. else
  2329. FLevel := AParentItem.Level + 1;
  2330. end;
  2331. destructor TJclPeResourceItem.Destroy;
  2332. begin
  2333. FreeAndNil(FList);
  2334. inherited Destroy;
  2335. end;
  2336. function TJclPeResourceItem.CompareName(AName: PChar): Boolean;
  2337. var
  2338. P: PChar;
  2339. begin
  2340. if IsName then
  2341. P := PChar(Name)
  2342. else
  2343. P := PChar(FEntry^.Name and $FFFF); // Integer encoded in a PChar
  2344. Result := CompareResourceName(AName, P);
  2345. end;
  2346. function TJclPeResourceItem.GetDataEntry: PImageResourceDataEntry;
  2347. begin
  2348. if GetIsDirectory then
  2349. Result := nil
  2350. else
  2351. Result := PImageResourceDataEntry(OffsetToRawData(FEntry^.OffsetToData));
  2352. end;
  2353. function TJclPeResourceItem.GetIsDirectory: Boolean;
  2354. begin
  2355. Result := FEntry^.OffsetToData and IMAGE_RESOURCE_DATA_IS_DIRECTORY <> 0;
  2356. end;
  2357. function TJclPeResourceItem.GetIsName: Boolean;
  2358. begin
  2359. Result := FEntry^.Name and IMAGE_RESOURCE_NAME_IS_STRING <> 0;
  2360. end;
  2361. function TJclPeResourceItem.GetLangID: LANGID;
  2362. begin
  2363. if IsDirectory then
  2364. begin
  2365. GetList;
  2366. if FList.Count = 1 then
  2367. Result := StrToIntDef(FList[0].Name, 0)
  2368. else
  2369. Result := 0;
  2370. end
  2371. else
  2372. Result := StrToIntDef(Name, 0);
  2373. end;
  2374. function TJclPeResourceItem.GetList: TJclPeResourceList;
  2375. begin
  2376. if not IsDirectory then
  2377. begin
  2378. if Image.NoExceptions then
  2379. begin
  2380. Result := nil;
  2381. Exit;
  2382. end
  2383. else
  2384. raise EJclPeImageError.CreateRes(@RsPeNotResDir);
  2385. end;
  2386. if FList = nil then
  2387. FList := FImage.ResourceListCreate(SubDirData, Self);
  2388. Result := FList;
  2389. end;
  2390. function TJclPeResourceItem.GetName: string;
  2391. begin
  2392. if IsName then
  2393. begin
  2394. if FNameCache = '' then
  2395. begin
  2396. with PImageResourceDirStringU(OffsetToRawData(FEntry^.Name))^ do
  2397. FNameCache := WideCharLenToString(NameString, Length);
  2398. StrResetLength(FNameCache);
  2399. end;
  2400. Result := FNameCache;
  2401. end
  2402. else
  2403. Result := IntToStr(FEntry^.Name and $FFFF);
  2404. end;
  2405. function TJclPeResourceItem.GetParameterName: string;
  2406. begin
  2407. if IsName then
  2408. Result := Name
  2409. else
  2410. Result := Format('#%d', [FEntry^.Name and $FFFF]);
  2411. end;
  2412. function TJclPeResourceItem.GetRawEntryData: Pointer;
  2413. begin
  2414. if GetIsDirectory then
  2415. Result := nil
  2416. else
  2417. Result := FImage.RvaToVa(GetDataEntry^.OffsetToData);
  2418. end;
  2419. function TJclPeResourceItem.GetRawEntryDataSize: Integer;
  2420. begin
  2421. if GetIsDirectory then
  2422. Result := -1
  2423. else
  2424. Result := PImageResourceDataEntry(OffsetToRawData(FEntry^.OffsetToData))^.Size;
  2425. end;
  2426. function TJclPeResourceItem.GetResourceType: TJclPeResourceKind;
  2427. begin
  2428. with Level1Item do
  2429. begin
  2430. if FEntry^.Name < Cardinal(High(TJclPeResourceKind)) then
  2431. Result := TJclPeResourceKind(FEntry^.Name)
  2432. else
  2433. Result := rtUserDefined
  2434. end;
  2435. end;
  2436. function TJclPeResourceItem.GetResourceTypeStr: string;
  2437. begin
  2438. with Level1Item do
  2439. begin
  2440. if FEntry^.Name < Cardinal(High(TJclPeResourceKind)) then
  2441. Result := Copy(GetEnumName(TypeInfo(TJclPeResourceKind), Ord(FEntry^.Name)), 3, 30)
  2442. else
  2443. Result := Name;
  2444. end;
  2445. end;
  2446. function TJclPeResourceItem.Level1Item: TJclPeResourceItem;
  2447. begin
  2448. Result := Self;
  2449. while Result.FParentItem <> nil do
  2450. Result := Result.FParentItem;
  2451. end;
  2452. function TJclPeResourceItem.OffsetToRawData(Ofs: DWORD): TJclAddr;
  2453. begin
  2454. Result := (Ofs and $7FFFFFFF) + Image.ResourceVA;
  2455. end;
  2456. function TJclPeResourceItem.SubDirData: PImageResourceDirectory;
  2457. begin
  2458. Result := Pointer(OffsetToRawData(FEntry^.OffsetToData));
  2459. end;
  2460. //=== { TJclPeResourceList } =================================================
  2461. constructor TJclPeResourceList.Create(AImage: TJclPeImage;
  2462. AParentItem: TJclPeResourceItem; ADirectory: PImageResourceDirectory);
  2463. begin
  2464. inherited Create(AImage);
  2465. FDirectory := ADirectory;
  2466. FParentItem := AParentItem;
  2467. CreateList(AParentItem);
  2468. end;
  2469. procedure TJclPeResourceList.CreateList(AParentItem: TJclPeResourceItem);
  2470. var
  2471. Entry: PImageResourceDirectoryEntry;
  2472. DirItem: TJclPeResourceItem;
  2473. I: Integer;
  2474. begin
  2475. if FDirectory = nil then
  2476. Exit;
  2477. Entry := Pointer(TJclAddr(FDirectory) + SizeOf(TImageResourceDirectory));
  2478. for I := 1 to DWORD(FDirectory^.NumberOfNamedEntries) + DWORD(FDirectory^.NumberOfIdEntries) do
  2479. begin
  2480. DirItem := Image.ResourceItemCreate(Entry, AParentItem);
  2481. Add(DirItem);
  2482. Inc(Entry);
  2483. end;
  2484. end;
  2485. function TJclPeResourceList.FindName(const Name: string): TJclPeResourceItem;
  2486. var
  2487. I: Integer;
  2488. begin
  2489. Result := nil;
  2490. for I := 0 to Count - 1 do
  2491. if StrSame(Items[I].Name, Name) then
  2492. begin
  2493. Result := Items[I];
  2494. Break;
  2495. end;
  2496. end;
  2497. function TJclPeResourceList.GetItems(Index: Integer): TJclPeResourceItem;
  2498. begin
  2499. Result := TJclPeResourceItem(Get(Index));
  2500. end;
  2501. //=== { TJclPeRootResourceList } =============================================
  2502. destructor TJclPeRootResourceList.Destroy;
  2503. begin
  2504. FreeAndNil(FManifestContent);
  2505. inherited Destroy;
  2506. end;
  2507. function TJclPeRootResourceList.FindResource(ResourceType: TJclPeResourceKind;
  2508. const ResourceName: string): TJclPeResourceItem;
  2509. var
  2510. I: Integer;
  2511. TypeItem: TJclPeResourceItem;
  2512. begin
  2513. Result := nil;
  2514. TypeItem := nil;
  2515. for I := 0 to Count - 1 do
  2516. begin
  2517. if Items[I].ResourceType = ResourceType then
  2518. begin
  2519. TypeItem := Items[I];
  2520. Break;
  2521. end;
  2522. end;
  2523. if TypeItem <> nil then
  2524. if ResourceName = '' then
  2525. Result := TypeItem
  2526. else
  2527. with TypeItem.List do
  2528. for I := 0 to Count - 1 do
  2529. if Items[I].Name = ResourceName then
  2530. begin
  2531. Result := Items[I];
  2532. Break;
  2533. end;
  2534. end;
  2535. function TJclPeRootResourceList.FindResource(const ResourceType: PChar;
  2536. const ResourceName: PChar): TJclPeResourceItem;
  2537. var
  2538. I: Integer;
  2539. TypeItem: TJclPeResourceItem;
  2540. begin
  2541. Result := nil;
  2542. TypeItem := nil;
  2543. for I := 0 to Count - 1 do
  2544. if Items[I].CompareName(ResourceType) then
  2545. begin
  2546. TypeItem := Items[I];
  2547. Break;
  2548. end;
  2549. if TypeItem <> nil then
  2550. if ResourceName = nil then
  2551. Result := TypeItem
  2552. else
  2553. with TypeItem.List do
  2554. for I := 0 to Count - 1 do
  2555. if Items[I].CompareName(ResourceName) then
  2556. begin
  2557. Result := Items[I];
  2558. Break;
  2559. end;
  2560. end;
  2561. function TJclPeRootResourceList.GetManifestContent: TStrings;
  2562. var
  2563. ManifestFileName: string;
  2564. ResItem: TJclPeResourceItem;
  2565. ResStream: TJclPeResourceRawStream;
  2566. begin
  2567. if FManifestContent = nil then
  2568. begin
  2569. FManifestContent := TStringList.Create;
  2570. ResItem := FindResource(RT_MANIFEST, CREATEPROCESS_MANIFEST_RESOURCE_ID);
  2571. if ResItem = nil then
  2572. begin
  2573. ManifestFileName := Image.FileName + MANIFESTExtension;
  2574. if FileExists(ManifestFileName) then
  2575. FManifestContent.LoadFromFile(ManifestFileName);
  2576. end
  2577. else
  2578. begin
  2579. ResStream := TJclPeResourceRawStream.Create(ResItem.List[0]);
  2580. try
  2581. FManifestContent.LoadFromStream(ResStream);
  2582. finally
  2583. ResStream.Free;
  2584. end;
  2585. end;
  2586. end;
  2587. Result := FManifestContent;
  2588. end;
  2589. function TJclPeRootResourceList.ListResourceNames(ResourceType: TJclPeResourceKind;
  2590. const Strings: TStrings): Boolean;
  2591. var
  2592. ResTypeItem, TempItem: TJclPeResourceItem;
  2593. I: Integer;
  2594. begin
  2595. ResTypeItem := FindResource(ResourceType, '');
  2596. Result := (ResTypeItem <> nil);
  2597. if Result then
  2598. begin
  2599. Strings.BeginUpdate;
  2600. try
  2601. with ResTypeItem.List do
  2602. for I := 0 to Count - 1 do
  2603. begin
  2604. TempItem := Items[I];
  2605. Strings.AddObject(TempItem.Name, Pointer(TempItem.IsName));
  2606. end;
  2607. finally
  2608. Strings.EndUpdate;
  2609. end;
  2610. end;
  2611. end;
  2612. //=== { TJclPeRelocEntry } ===================================================
  2613. constructor TJclPeRelocEntry.Create(AChunk: PImageBaseRelocation; ACount: Integer);
  2614. begin
  2615. inherited Create;
  2616. FChunk := AChunk;
  2617. FCount := ACount;
  2618. end;
  2619. function TJclPeRelocEntry.GetRelocations(Index: Integer): TJclPeRelocation;
  2620. var
  2621. Temp: Word;
  2622. begin
  2623. Temp := PWord(TJclAddr(FChunk) + SizeOf(TImageBaseRelocation) + DWORD(Index) * SizeOf(Word))^;
  2624. Result.Address := Temp and $0FFF;
  2625. Result.RelocType := (Temp and $F000) shr 12;
  2626. Result.VirtualAddress := TJclAddr(Result.Address) + VirtualAddress;
  2627. end;
  2628. function TJclPeRelocEntry.GetSize: DWORD;
  2629. begin
  2630. Result := FChunk^.SizeOfBlock;
  2631. end;
  2632. function TJclPeRelocEntry.GetVirtualAddress: DWORD;
  2633. begin
  2634. Result := FChunk^.VirtualAddress;
  2635. end;
  2636. //=== { TJclPeRelocList } ====================================================
  2637. constructor TJclPeRelocList.Create(AImage: TJclPeImage);
  2638. begin
  2639. inherited Create(AImage);
  2640. CreateList;
  2641. end;
  2642. procedure TJclPeRelocList.CreateList;
  2643. var
  2644. Chunk: PImageBaseRelocation;
  2645. Item: TJclPeRelocEntry;
  2646. RelocCount: Integer;
  2647. begin
  2648. with Image do
  2649. begin
  2650. if not StatusOK then
  2651. Exit;
  2652. Chunk := DirectoryEntryToData(IMAGE_DIRECTORY_ENTRY_BASERELOC);
  2653. if Chunk = nil then
  2654. Exit;
  2655. FAllItemCount := 0;
  2656. while Chunk^.SizeOfBlock <> 0 do
  2657. begin
  2658. RelocCount := (Chunk^.SizeOfBlock - SizeOf(TImageBaseRelocation)) div SizeOf(Word);
  2659. Item := TJclPeRelocEntry.Create(Chunk, RelocCount);
  2660. Inc(FAllItemCount, RelocCount);
  2661. Add(Item);
  2662. Chunk := Pointer(TJclAddr(Chunk) + Chunk^.SizeOfBlock);
  2663. end;
  2664. end;
  2665. end;
  2666. function TJclPeRelocList.GetAllItems(Index: Integer): TJclPeRelocation;
  2667. var
  2668. I, N, C: Integer;
  2669. begin
  2670. N := Index;
  2671. for I := 0 to Count - 1 do
  2672. begin
  2673. C := Items[I].Count;
  2674. Dec(N, C);
  2675. if N < 0 then
  2676. begin
  2677. Result := Items[I][N + C];
  2678. Break;
  2679. end;
  2680. end;
  2681. end;
  2682. function TJclPeRelocList.GetItems(Index: Integer): TJclPeRelocEntry;
  2683. begin
  2684. Result := TJclPeRelocEntry(Get(Index));
  2685. end;
  2686. //=== { TJclPeDebugList } ====================================================
  2687. constructor TJclPeDebugList.Create(AImage: TJclPeImage);
  2688. begin
  2689. inherited Create(AImage);
  2690. OwnsObjects := False;
  2691. CreateList;
  2692. end;
  2693. procedure TJclPeDebugList.CreateList;
  2694. var
  2695. DebugImageDir: TImageDataDirectory;
  2696. DebugDir: PImageDebugDirectory;
  2697. Header: PImageSectionHeader;
  2698. FormatCount, I: Integer;
  2699. begin
  2700. with Image do
  2701. begin
  2702. if not StatusOK then
  2703. Exit;
  2704. DebugImageDir := Directories[IMAGE_DIRECTORY_ENTRY_DEBUG];
  2705. if DebugImageDir.VirtualAddress = 0 then
  2706. Exit;
  2707. if GetSectionHeader(DebugSectionName, Header) and
  2708. (Header^.VirtualAddress = DebugImageDir.VirtualAddress) then
  2709. begin
  2710. FormatCount := DebugImageDir.Size;
  2711. DebugDir := RvaToVa(Header^.VirtualAddress);
  2712. end
  2713. else
  2714. begin
  2715. if not GetSectionHeader(ReadOnlySectionName, Header) then
  2716. Exit;
  2717. FormatCount := DebugImageDir.Size div SizeOf(TImageDebugDirectory);
  2718. DebugDir := Pointer(MappedAddress + DebugImageDir.VirtualAddress -
  2719. Header^.VirtualAddress + Header^.PointerToRawData);
  2720. end;
  2721. for I := 1 to FormatCount do
  2722. begin
  2723. Add(TObject(DebugDir));
  2724. Inc(DebugDir);
  2725. end;
  2726. end;
  2727. end;
  2728. function TJclPeDebugList.GetItems(Index: Integer): TImageDebugDirectory;
  2729. begin
  2730. Result := PImageDebugDirectory(Get(Index))^;
  2731. end;
  2732. //=== { TJclPeCertificate } ==================================================
  2733. constructor TJclPeCertificate.Create(AHeader: TWinCertificate; AData: Pointer);
  2734. begin
  2735. inherited Create;
  2736. FHeader := AHeader;
  2737. FData := AData;
  2738. end;
  2739. //=== { TJclPeCertificateList } ==============================================
  2740. constructor TJclPeCertificateList.Create(AImage: TJclPeImage);
  2741. begin
  2742. inherited Create(AImage);
  2743. CreateList;
  2744. end;
  2745. procedure TJclPeCertificateList.CreateList;
  2746. var
  2747. Directory: TImageDataDirectory;
  2748. CertPtr: PChar;
  2749. TotalSize: Integer;
  2750. Item: TJclPeCertificate;
  2751. begin
  2752. Directory := Image.Directories[IMAGE_DIRECTORY_ENTRY_SECURITY];
  2753. if Directory.VirtualAddress = 0 then
  2754. Exit;
  2755. CertPtr := Image.RawToVa(Directory.VirtualAddress); // Security directory is a raw offset
  2756. TotalSize := Directory.Size;
  2757. while TotalSize >= SizeOf(TWinCertificate) do
  2758. begin
  2759. Item := TJclPeCertificate.Create(PWinCertificate(CertPtr)^, CertPtr + SizeOf(TWinCertificate));
  2760. Dec(TotalSize, Item.Header.dwLength);
  2761. Add(Item);
  2762. end;
  2763. end;
  2764. function TJclPeCertificateList.GetItems(Index: Integer): TJclPeCertificate;
  2765. begin
  2766. Result := TJclPeCertificate(Get(Index));
  2767. end;
  2768. //=== { TJclPeCLRHeader } ====================================================
  2769. constructor TJclPeCLRHeader.Create(AImage: TJclPeImage);
  2770. begin
  2771. FImage := AImage;
  2772. ReadHeader;
  2773. end;
  2774. function TJclPeCLRHeader.GetHasMetadata: Boolean;
  2775. const
  2776. METADATA_SIGNATURE = $424A5342; // Reference: Partition II Metadata.doc - 23.2.1 Metadata root
  2777. begin
  2778. with Header.MetaData do
  2779. Result := (VirtualAddress <> 0) and (PDWORD(FImage.RvaToVa(VirtualAddress))^ = METADATA_SIGNATURE);
  2780. end;
  2781. { TODO -cDOC : "Flier Lu" <flier_lu att yahoo dott com dott cn> }
  2782. function TJclPeCLRHeader.GetVersionString: string;
  2783. begin
  2784. Result := FormatVersionString(Header.MajorRuntimeVersion, Header.MinorRuntimeVersion);
  2785. end;
  2786. procedure TJclPeCLRHeader.ReadHeader;
  2787. var
  2788. HeaderPtr: PImageCor20Header;
  2789. begin
  2790. HeaderPtr := Image.DirectoryEntryToData(IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR);
  2791. if (HeaderPtr <> nil) and (HeaderPtr^.cb >= SizeOf(TImageCor20Header)) then
  2792. FHeader := HeaderPtr^;
  2793. end;
  2794. //=== { TJclPeImage } ========================================================
  2795. constructor TJclPeImage.Create(ANoExceptions: Boolean);
  2796. begin
  2797. FNoExceptions := ANoExceptions;
  2798. FReadOnlyAccess := True;
  2799. FImageSections := TStringList.Create;
  2800. FStringTable := TStringList.Create;
  2801. end;
  2802. destructor TJclPeImage.Destroy;
  2803. begin
  2804. Clear;
  2805. FreeAndNil(FImageSections);
  2806. FStringTable.Free;
  2807. inherited Destroy;
  2808. end;
  2809. procedure TJclPeImage.AfterOpen;
  2810. begin
  2811. end;
  2812. procedure TJclPeImage.AttachLoadedModule(const Handle: HMODULE);
  2813. procedure AttachLoadedModule32;
  2814. var
  2815. NtHeaders: PImageNtHeaders32;
  2816. begin
  2817. NtHeaders := PeMapImgNtHeaders32(Pointer(Handle));
  2818. if NtHeaders = nil then
  2819. FStatus := stNotPE
  2820. else
  2821. begin
  2822. FStatus := stOk;
  2823. FAttachedImage := True;
  2824. FFileName := GetModulePath(Handle);
  2825. // OF: possible loss of data
  2826. FLoadedImage.ModuleName := PAnsiChar(AnsiString(FFileName));
  2827. FLoadedImage.hFile := INVALID_HANDLE_VALUE;
  2828. FLoadedImage.MappedAddress := Pointer(Handle);
  2829. FLoadedImage.FileHeader := PImageNtHeaders(NtHeaders);
  2830. FLoadedImage.NumberOfSections := NtHeaders^.FileHeader.NumberOfSections;
  2831. FLoadedImage.Sections := PeMapImgSections32(NtHeaders);
  2832. FLoadedImage.LastRvaSection := FLoadedImage.Sections;
  2833. FLoadedImage.Characteristics := NtHeaders^.FileHeader.Characteristics;
  2834. FLoadedImage.fSystemImage := (FLoadedImage.Characteristics and IMAGE_FILE_SYSTEM <> 0);
  2835. FLoadedImage.fDOSImage := False;
  2836. FLoadedImage.SizeOfImage := NtHeaders^.OptionalHeader.SizeOfImage;
  2837. ReadImageSections;
  2838. ReadStringTable;
  2839. AfterOpen;
  2840. end;
  2841. RaiseStatusException;
  2842. end;
  2843. procedure AttachLoadedModule64;
  2844. var
  2845. NtHeaders: PImageNtHeaders64;
  2846. begin
  2847. NtHeaders := PeMapImgNtHeaders64(Pointer(Handle));
  2848. if NtHeaders = nil then
  2849. FStatus := stNotPE
  2850. else
  2851. begin
  2852. FStatus := stOk;
  2853. FAttachedImage := True;
  2854. FFileName := GetModulePath(Handle);
  2855. // OF: possible loss of data
  2856. FLoadedImage.ModuleName := PAnsiChar(AnsiString(FFileName));
  2857. FLoadedImage.hFile := INVALID_HANDLE_VALUE;
  2858. FLoadedImage.MappedAddress := Pointer(Handle);
  2859. FLoadedImage.FileHeader := PImageNtHeaders(NtHeaders);
  2860. FLoadedImage.NumberOfSections := NtHeaders^.FileHeader.NumberOfSections;
  2861. FLoadedImage.Sections := PeMapImgSections64(NtHeaders);
  2862. FLoadedImage.LastRvaSection := FLoadedImage.Sections;
  2863. FLoadedImage.Characteristics := NtHeaders^.FileHeader.Characteristics;
  2864. FLoadedImage.fSystemImage := (FLoadedImage.Characteristics and IMAGE_FILE_SYSTEM <> 0);
  2865. FLoadedImage.fDOSImage := False;
  2866. FLoadedImage.SizeOfImage := NtHeaders^.OptionalHeader.SizeOfImage;
  2867. ReadImageSections;
  2868. ReadStringTable;
  2869. AfterOpen;
  2870. end;
  2871. RaiseStatusException;
  2872. end;
  2873. begin
  2874. Clear;
  2875. if Handle = 0 then
  2876. Exit;
  2877. FTarget := PeMapImgTarget(Pointer(Handle));
  2878. case Target of
  2879. taWin32:
  2880. AttachLoadedModule32;
  2881. taWin64:
  2882. AttachLoadedModule64;
  2883. taUnknown:
  2884. FStatus := stNotSupported;
  2885. end;
  2886. end;
  2887. function TJclPeImage.CalculateCheckSum: DWORD;
  2888. var
  2889. C: DWORD;
  2890. begin
  2891. if StatusOK then
  2892. begin
  2893. CheckNotAttached;
  2894. if CheckSumMappedFile(FLoadedImage.MappedAddress, FLoadedImage.SizeOfImage,
  2895. C, Result) = nil then
  2896. RaiseLastOSError;
  2897. end
  2898. else
  2899. Result := 0;
  2900. end;
  2901. procedure TJclPeImage.CheckNotAttached;
  2902. begin
  2903. if FAttachedImage then
  2904. raise EJclPeImageError.CreateRes(@RsPeNotAvailableForAttached);
  2905. end;
  2906. procedure TJclPeImage.Clear;
  2907. begin
  2908. FImageSections.Clear;
  2909. FStringTable.Clear;
  2910. FreeAndNil(FCertificateList);
  2911. FreeAndNil(FCLRHeader);
  2912. FreeAndNil(FDebugList);
  2913. FreeAndNil(FImportList);
  2914. FreeAndNil(FExportList);
  2915. FreeAndNil(FRelocationList);
  2916. FreeAndNil(FResourceList);
  2917. FreeAndNil(FVersionInfo);
  2918. if not FAttachedImage and StatusOK then
  2919. UnMapAndLoad(FLoadedImage);
  2920. ResetMemory(FLoadedImage, SizeOf(FLoadedImage));
  2921. FStatus := stNotLoaded;
  2922. FAttachedImage := False;
  2923. end;
  2924. class function TJclPeImage.DateTimeToStamp(const DateTime: TDateTime): DWORD;
  2925. begin
  2926. Result := Round((DateTime - UnixTimeStart) * SecsPerDay);
  2927. end;
  2928. class function TJclPeImage.DebugTypeNames(DebugType: DWORD): string;
  2929. begin
  2930. case DebugType of
  2931. IMAGE_DEBUG_TYPE_UNKNOWN:
  2932. Result := LoadResString(@RsPeDEBUG_UNKNOWN);
  2933. IMAGE_DEBUG_TYPE_COFF:
  2934. Result := LoadResString(@RsPeDEBUG_COFF);
  2935. IMAGE_DEBUG_TYPE_CODEVIEW:
  2936. Result := LoadResString(@RsPeDEBUG_CODEVIEW);
  2937. IMAGE_DEBUG_TYPE_FPO:
  2938. Result := LoadResString(@RsPeDEBUG_FPO);
  2939. IMAGE_DEBUG_TYPE_MISC:
  2940. Result := LoadResString(@RsPeDEBUG_MISC);
  2941. IMAGE_DEBUG_TYPE_EXCEPTION:
  2942. Result := LoadResString(@RsPeDEBUG_EXCEPTION);
  2943. IMAGE_DEBUG_TYPE_FIXUP:
  2944. Result := LoadResString(@RsPeDEBUG_FIXUP);
  2945. IMAGE_DEBUG_TYPE_OMAP_TO_SRC:
  2946. Result := LoadResString(@RsPeDEBUG_OMAP_TO_SRC);
  2947. IMAGE_DEBUG_TYPE_OMAP_FROM_SRC:
  2948. Result := LoadResString(@RsPeDEBUG_OMAP_FROM_SRC);
  2949. else
  2950. Result := LoadResString(@RsPeDEBUG_UNKNOWN);
  2951. end;
  2952. end;
  2953. function TJclPeImage.DirectoryEntryToData(Directory: Word): Pointer;
  2954. var
  2955. Size: DWORD;
  2956. begin
  2957. Size := 0;
  2958. Result := ImageDirectoryEntryToData(FLoadedImage.MappedAddress, FAttachedImage, Directory, Size);
  2959. end;
  2960. class function TJclPeImage.DirectoryNames(Directory: Word): string;
  2961. begin
  2962. case Directory of
  2963. IMAGE_DIRECTORY_ENTRY_EXPORT:
  2964. Result := LoadResString(@RsPeImg_00);
  2965. IMAGE_DIRECTORY_ENTRY_IMPORT:
  2966. Result := LoadResString(@RsPeImg_01);
  2967. IMAGE_DIRECTORY_ENTRY_RESOURCE:
  2968. Result := LoadResString(@RsPeImg_02);
  2969. IMAGE_DIRECTORY_ENTRY_EXCEPTION:
  2970. Result := LoadResString(@RsPeImg_03);
  2971. IMAGE_DIRECTORY_ENTRY_SECURITY:
  2972. Result := LoadResString(@RsPeImg_04);
  2973. IMAGE_DIRECTORY_ENTRY_BASERELOC:
  2974. Result := LoadResString(@RsPeImg_05);
  2975. IMAGE_DIRECTORY_ENTRY_DEBUG:
  2976. Result := LoadResString(@RsPeImg_06);
  2977. IMAGE_DIRECTORY_ENTRY_COPYRIGHT:
  2978. Result := LoadResString(@RsPeImg_07);
  2979. IMAGE_DIRECTORY_ENTRY_GLOBALPTR:
  2980. Result := LoadResString(@RsPeImg_08);
  2981. IMAGE_DIRECTORY_ENTRY_TLS:
  2982. Result := LoadResString(@RsPeImg_09);
  2983. IMAGE_DIRECTORY_ENTRY_LOAD_CONFIG:
  2984. Result := LoadResString(@RsPeImg_10);
  2985. IMAGE_DIRECTORY_ENTRY_BOUND_IMPORT:
  2986. Result := LoadResString(@RsPeImg_11);
  2987. IMAGE_DIRECTORY_ENTRY_IAT:
  2988. Result := LoadResString(@RsPeImg_12);
  2989. IMAGE_DIRECTORY_ENTRY_DELAY_IMPORT:
  2990. Result := LoadResString(@RsPeImg_13);
  2991. IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR:
  2992. Result := LoadResString(@RsPeImg_14);
  2993. else
  2994. Result := Format(LoadResString(@RsPeImg_Reserved), [Directory]);
  2995. end;
  2996. end;
  2997. class function TJclPeImage.ExpandBySearchPath(const ModuleName, BasePath: string): TFileName;
  2998. var
  2999. FullName: array [0..MAX_PATH] of Char;
  3000. FilePart: PChar;
  3001. begin
  3002. Result := PathAddSeparator(ExtractFilePath(BasePath)) + ModuleName;
  3003. if FileExists(Result) then
  3004. Exit;
  3005. FilePart := nil;
  3006. if SearchPath(nil, PChar(ModuleName), nil, Length(FullName), FullName, FilePart) = 0 then
  3007. Result := ModuleName
  3008. else
  3009. Result := FullName;
  3010. end;
  3011. function TJclPeImage.ExpandModuleName(const ModuleName: string): TFileName;
  3012. begin
  3013. Result := ExpandBySearchPath(ModuleName, ExtractFilePath(FFileName));
  3014. end;
  3015. function TJclPeImage.GetCertificateList: TJclPeCertificateList;
  3016. begin
  3017. if FCertificateList = nil then
  3018. FCertificateList := TJclPeCertificateList.Create(Self);
  3019. Result := FCertificateList;
  3020. end;
  3021. function TJclPeImage.GetCLRHeader: TJclPeCLRHeader;
  3022. begin
  3023. if FCLRHeader = nil then
  3024. FCLRHeader := TJclPeCLRHeader.Create(Self);
  3025. Result := FCLRHeader;
  3026. end;
  3027. function TJclPeImage.GetDebugList: TJclPeDebugList;
  3028. begin
  3029. if FDebugList = nil then
  3030. FDebugList := TJclPeDebugList.Create(Self);
  3031. Result := FDebugList;
  3032. end;
  3033. function TJclPeImage.GetDescription: string;
  3034. var
  3035. UTF8DescriptionName: TUTF8String;
  3036. begin
  3037. if DirectoryExists[IMAGE_DIRECTORY_ENTRY_COPYRIGHT] then
  3038. begin
  3039. UTF8DescriptionName := PAnsiChar(DirectoryEntryToData(IMAGE_DIRECTORY_ENTRY_COPYRIGHT));
  3040. if not TryUTF8ToString(UTF8DescriptionName, Result) then
  3041. Result := string(UTF8DescriptionName);
  3042. end
  3043. else
  3044. Result := '';
  3045. end;
  3046. function TJclPeImage.GetDirectories(Directory: Word): TImageDataDirectory;
  3047. begin
  3048. if StatusOK then
  3049. begin
  3050. case Target of
  3051. taWin32:
  3052. Result := PImageNtHeaders32(FLoadedImage.FileHeader)^.OptionalHeader.DataDirectory[Directory];
  3053. taWin64:
  3054. Result := PImageNtHeaders64(FLoadedImage.FileHeader)^.OptionalHeader.DataDirectory[Directory];
  3055. else
  3056. Result.VirtualAddress := 0;
  3057. Result.Size := 0;
  3058. end
  3059. end
  3060. else
  3061. begin
  3062. Result.VirtualAddress := 0;
  3063. Result.Size := 0;
  3064. end;
  3065. end;
  3066. function TJclPeImage.GetDirectoryExists(Directory: Word): Boolean;
  3067. begin
  3068. Result := (Directories[Directory].VirtualAddress <> 0);
  3069. end;
  3070. function TJclPeImage.GetExportList: TJclPeExportFuncList;
  3071. begin
  3072. if FExportList = nil then
  3073. FExportList := TJclPeExportFuncList.Create(Self);
  3074. Result := FExportList;
  3075. end;
  3076. function TJclPeImage.GetFileProperties: TJclPeFileProperties;
  3077. var
  3078. FileAttributesEx: WIN32_FILE_ATTRIBUTE_DATA;
  3079. Size: TJclULargeInteger;
  3080. begin
  3081. ResetMemory(Result, SizeOf(Result));
  3082. if GetFileAttributesEx(PChar(FileName), GetFileExInfoStandard, @FileAttributesEx) then
  3083. begin
  3084. Size.LowPart := FileAttributesEx.nFileSizeLow;
  3085. Size.HighPart := FileAttributesEx.nFileSizeHigh;
  3086. Result.Size := Size.QuadPart;
  3087. Result.CreationTime := FileTimeToLocalDateTime(FileAttributesEx.ftCreationTime);
  3088. Result.LastAccessTime := FileTimeToLocalDateTime(FileAttributesEx.ftLastAccessTime);
  3089. Result.LastWriteTime := FileTimeToLocalDateTime(FileAttributesEx.ftLastWriteTime);
  3090. Result.Attributes := FileAttributesEx.dwFileAttributes;
  3091. end;
  3092. end;
  3093. function TJclPeImage.GetHeaderValues(Index: TJclPeHeader): string;
  3094. function GetMachineString(Value: DWORD): string;
  3095. begin
  3096. case Value of
  3097. IMAGE_FILE_MACHINE_UNKNOWN:
  3098. Result := LoadResString(@RsPeMACHINE_UNKNOWN);
  3099. IMAGE_FILE_MACHINE_I386:
  3100. Result := LoadResString(@RsPeMACHINE_I386);
  3101. IMAGE_FILE_MACHINE_R3000:
  3102. Result := LoadResString(@RsPeMACHINE_R3000);
  3103. IMAGE_FILE_MACHINE_R4000:
  3104. Result := LoadResString(@RsPeMACHINE_R4000);
  3105. IMAGE_FILE_MACHINE_R10000:
  3106. Result := LoadResString(@RsPeMACHINE_R10000);
  3107. IMAGE_FILE_MACHINE_WCEMIPSV2:
  3108. Result := LoadResString(@RsPeMACHINE_WCEMIPSV2);
  3109. IMAGE_FILE_MACHINE_ALPHA:
  3110. Result := LoadResString(@RsPeMACHINE_ALPHA);
  3111. IMAGE_FILE_MACHINE_SH3:
  3112. Result := LoadResString(@RsPeMACHINE_SH3); // SH3 little-endian
  3113. IMAGE_FILE_MACHINE_SH3DSP:
  3114. Result := LoadResString(@RsPeMACHINE_SH3DSP);
  3115. IMAGE_FILE_MACHINE_SH3E:
  3116. Result := LoadResString(@RsPeMACHINE_SH3E); // SH3E little-endian
  3117. IMAGE_FILE_MACHINE_SH4:
  3118. Result := LoadResString(@RsPeMACHINE_SH4); // SH4 little-endian
  3119. IMAGE_FILE_MACHINE_SH5:
  3120. Result := LoadResString(@RsPeMACHINE_SH5); // SH5
  3121. IMAGE_FILE_MACHINE_ARM:
  3122. Result := LoadResString(@RsPeMACHINE_ARM); // ARM Little-Endian
  3123. IMAGE_FILE_MACHINE_THUMB:
  3124. Result := LoadResString(@RsPeMACHINE_THUMB);
  3125. IMAGE_FILE_MACHINE_AM33:
  3126. Result := LoadResString(@RsPeMACHINE_AM33);
  3127. IMAGE_FILE_MACHINE_POWERPC:
  3128. Result := LoadResString(@RsPeMACHINE_POWERPC);
  3129. IMAGE_FILE_MACHINE_POWERPCFP:
  3130. Result := LoadResString(@RsPeMACHINE_POWERPCFP);
  3131. IMAGE_FILE_MACHINE_IA64:
  3132. Result := LoadResString(@RsPeMACHINE_IA64); // Intel 64
  3133. IMAGE_FILE_MACHINE_MIPS16:
  3134. Result := LoadResString(@RsPeMACHINE_MIPS16); // MIPS
  3135. IMAGE_FILE_MACHINE_ALPHA64:
  3136. Result := LoadResString(@RsPeMACHINE_AMPHA64); // ALPHA64
  3137. //IMAGE_FILE_MACHINE_AXP64
  3138. IMAGE_FILE_MACHINE_MIPSFPU:
  3139. Result := LoadResString(@RsPeMACHINE_MIPSFPU); // MIPS
  3140. IMAGE_FILE_MACHINE_MIPSFPU16:
  3141. Result := LoadResString(@RsPeMACHINE_MIPSFPU16); // MIPS
  3142. IMAGE_FILE_MACHINE_TRICORE:
  3143. Result := LoadResString(@RsPeMACHINE_TRICORE); // Infineon
  3144. IMAGE_FILE_MACHINE_CEF:
  3145. Result := LoadResString(@RsPeMACHINE_CEF);
  3146. IMAGE_FILE_MACHINE_EBC:
  3147. Result := LoadResString(@RsPeMACHINE_EBC); // EFI Byte Code
  3148. IMAGE_FILE_MACHINE_AMD64:
  3149. Result := LoadResString(@RsPeMACHINE_AMD64); // AMD64 (K8)
  3150. IMAGE_FILE_MACHINE_M32R:
  3151. Result := LoadResString(@RsPeMACHINE_M32R); // M32R little-endian
  3152. IMAGE_FILE_MACHINE_CEE:
  3153. Result := LoadResString(@RsPeMACHINE_CEE);
  3154. else
  3155. Result := Format('[%.8x]', [Value]);
  3156. end;
  3157. end;
  3158. function GetSubsystemString(Value: DWORD): string;
  3159. begin
  3160. case Value of
  3161. IMAGE_SUBSYSTEM_UNKNOWN:
  3162. Result := LoadResString(@RsPeSUBSYSTEM_UNKNOWN);
  3163. IMAGE_SUBSYSTEM_NATIVE:
  3164. Result := LoadResString(@RsPeSUBSYSTEM_NATIVE);
  3165. IMAGE_SUBSYSTEM_WINDOWS_GUI:
  3166. Result := LoadResString(@RsPeSUBSYSTEM_WINDOWS_GUI);
  3167. IMAGE_SUBSYSTEM_WINDOWS_CUI:
  3168. Result := LoadResString(@RsPeSUBSYSTEM_WINDOWS_CUI);
  3169. IMAGE_SUBSYSTEM_OS2_CUI:
  3170. Result := LoadResString(@RsPeSUBSYSTEM_OS2_CUI);
  3171. IMAGE_SUBSYSTEM_POSIX_CUI:
  3172. Result := LoadResString(@RsPeSUBSYSTEM_POSIX_CUI);
  3173. IMAGE_SUBSYSTEM_RESERVED8:
  3174. Result := LoadResString(@RsPeSUBSYSTEM_RESERVED8);
  3175. else
  3176. Result := Format('[%.8x]', [Value]);
  3177. end;
  3178. end;
  3179. function GetHeaderValues32(Index: TJclPeHeader): string;
  3180. var
  3181. OptionalHeader: TImageOptionalHeader32;
  3182. begin
  3183. OptionalHeader := OptionalHeader32;
  3184. case Index of
  3185. JclPeHeader_Magic:
  3186. Result := IntToHex(OptionalHeader.Magic, 4);
  3187. JclPeHeader_LinkerVersion:
  3188. Result := FormatVersionString(OptionalHeader.MajorLinkerVersion, OptionalHeader.MinorLinkerVersion);
  3189. JclPeHeader_SizeOfCode:
  3190. Result := IntToHex(OptionalHeader.SizeOfCode, 8);
  3191. JclPeHeader_SizeOfInitializedData:
  3192. Result := IntToHex(OptionalHeader.SizeOfInitializedData, 8);
  3193. JclPeHeader_SizeOfUninitializedData:
  3194. Result := IntToHex(OptionalHeader.SizeOfUninitializedData, 8);
  3195. JclPeHeader_AddressOfEntryPoint:
  3196. Result := IntToHex(OptionalHeader.AddressOfEntryPoint, 8);
  3197. JclPeHeader_BaseOfCode:
  3198. Result := IntToHex(OptionalHeader.BaseOfCode, 8);
  3199. JclPeHeader_BaseOfData:
  3200. {$IFDEF DELPHI64_TEMPORARY}
  3201. System.Error(rePlatformNotImplemented);
  3202. {$ELSE ~DELPHI64_TEMPORARY}
  3203. Result := IntToHex(OptionalHeader.BaseOfData, 8);
  3204. {$ENDIF ~DELPHI64_TEMPORARY}
  3205. JclPeHeader_ImageBase:
  3206. Result := IntToHex(OptionalHeader.ImageBase, 8);
  3207. JclPeHeader_SectionAlignment:
  3208. Result := IntToHex(OptionalHeader.SectionAlignment, 8);
  3209. JclPeHeader_FileAlignment:
  3210. Result := IntToHex(OptionalHeader.FileAlignment, 8);
  3211. JclPeHeader_OperatingSystemVersion:
  3212. Result := FormatVersionString(OptionalHeader.MajorOperatingSystemVersion, OptionalHeader.MinorOperatingSystemVersion);
  3213. JclPeHeader_ImageVersion:
  3214. Result := FormatVersionString(OptionalHeader.MajorImageVersion, OptionalHeader.MinorImageVersion);
  3215. JclPeHeader_SubsystemVersion:
  3216. Result := FormatVersionString(OptionalHeader.MajorSubsystemVersion, OptionalHeader.MinorSubsystemVersion);
  3217. JclPeHeader_Win32VersionValue:
  3218. Result := IntToHex(OptionalHeader.Win32VersionValue, 8);
  3219. JclPeHeader_SizeOfImage:
  3220. Result := IntToHex(OptionalHeader.SizeOfImage, 8);
  3221. JclPeHeader_SizeOfHeaders:
  3222. Result := IntToHex(OptionalHeader.SizeOfHeaders, 8);
  3223. JclPeHeader_CheckSum:
  3224. Result := IntToHex(OptionalHeader.CheckSum, 8);
  3225. JclPeHeader_Subsystem:
  3226. Result := GetSubsystemString(OptionalHeader.Subsystem);
  3227. JclPeHeader_DllCharacteristics:
  3228. Result := IntToHex(OptionalHeader.DllCharacteristics, 4);
  3229. JclPeHeader_SizeOfStackReserve:
  3230. Result := IntToHex(OptionalHeader.SizeOfStackReserve, 8);
  3231. JclPeHeader_SizeOfStackCommit:
  3232. Result := IntToHex(OptionalHeader.SizeOfStackCommit, 8);
  3233. JclPeHeader_SizeOfHeapReserve:
  3234. Result := IntToHex(OptionalHeader.SizeOfHeapReserve, 8);
  3235. JclPeHeader_SizeOfHeapCommit:
  3236. Result := IntToHex(OptionalHeader.SizeOfHeapCommit, 8);
  3237. JclPeHeader_LoaderFlags:
  3238. Result := IntToHex(OptionalHeader.LoaderFlags, 8);
  3239. JclPeHeader_NumberOfRvaAndSizes:
  3240. Result := IntToHex(OptionalHeader.NumberOfRvaAndSizes, 8);
  3241. end;
  3242. end;
  3243. function GetHeaderValues64(Index: TJclPeHeader): string;
  3244. var
  3245. OptionalHeader: TImageOptionalHeader64;
  3246. begin
  3247. OptionalHeader := OptionalHeader64;
  3248. case Index of
  3249. JclPeHeader_Magic:
  3250. Result := IntToHex(OptionalHeader.Magic, 4);
  3251. JclPeHeader_LinkerVersion:
  3252. Result := FormatVersionString(OptionalHeader.MajorLinkerVersion, OptionalHeader.MinorLinkerVersion);
  3253. JclPeHeader_SizeOfCode:
  3254. Result := IntToHex(OptionalHeader.SizeOfCode, 8);
  3255. JclPeHeader_SizeOfInitializedData:
  3256. Result := IntToHex(OptionalHeader.SizeOfInitializedData, 8);
  3257. JclPeHeader_SizeOfUninitializedData:
  3258. Result := IntToHex(OptionalHeader.SizeOfUninitializedData, 8);
  3259. JclPeHeader_AddressOfEntryPoint:
  3260. Result := IntToHex(OptionalHeader.AddressOfEntryPoint, 8);
  3261. JclPeHeader_BaseOfCode:
  3262. Result := IntToHex(OptionalHeader.BaseOfCode, 8);
  3263. JclPeHeader_BaseOfData:
  3264. Result := ''; // IntToHex(OptionalHeader.BaseOfData, 8);
  3265. JclPeHeader_ImageBase:
  3266. Result := IntToHex(OptionalHeader.ImageBase, 16);
  3267. JclPeHeader_SectionAlignment:
  3268. Result := IntToHex(OptionalHeader.SectionAlignment, 8);
  3269. JclPeHeader_FileAlignment:
  3270. Result := IntToHex(OptionalHeader.FileAlignment, 8);
  3271. JclPeHeader_OperatingSystemVersion:
  3272. Result := FormatVersionString(OptionalHeader.MajorOperatingSystemVersion, OptionalHeader.MinorOperatingSystemVersion);
  3273. JclPeHeader_ImageVersion:
  3274. Result := FormatVersionString(OptionalHeader.MajorImageVersion, OptionalHeader.MinorImageVersion);
  3275. JclPeHeader_SubsystemVersion:
  3276. Result := FormatVersionString(OptionalHeader.MajorSubsystemVersion, OptionalHeader.MinorSubsystemVersion);
  3277. JclPeHeader_Win32VersionValue:
  3278. Result := IntToHex(OptionalHeader.Win32VersionValue, 8);
  3279. JclPeHeader_SizeOfImage:
  3280. Result := IntToHex(OptionalHeader.SizeOfImage, 8);
  3281. JclPeHeader_SizeOfHeaders:
  3282. Result := IntToHex(OptionalHeader.SizeOfHeaders, 8);
  3283. JclPeHeader_CheckSum:
  3284. Result := IntToHex(OptionalHeader.CheckSum, 8);
  3285. JclPeHeader_Subsystem:
  3286. Result := GetSubsystemString(OptionalHeader.Subsystem);
  3287. JclPeHeader_DllCharacteristics:
  3288. Result := IntToHex(OptionalHeader.DllCharacteristics, 4);
  3289. JclPeHeader_SizeOfStackReserve:
  3290. Result := IntToHex(OptionalHeader.SizeOfStackReserve, 16);
  3291. JclPeHeader_SizeOfStackCommit:
  3292. Result := IntToHex(OptionalHeader.SizeOfStackCommit, 16);
  3293. JclPeHeader_SizeOfHeapReserve:
  3294. Result := IntToHex(OptionalHeader.SizeOfHeapReserve, 16);
  3295. JclPeHeader_SizeOfHeapCommit:
  3296. Result := IntToHex(OptionalHeader.SizeOfHeapCommit, 16);
  3297. JclPeHeader_LoaderFlags:
  3298. Result := IntToHex(OptionalHeader.LoaderFlags, 8);
  3299. JclPeHeader_NumberOfRvaAndSizes:
  3300. Result := IntToHex(OptionalHeader.NumberOfRvaAndSizes, 8);
  3301. end;
  3302. end;
  3303. begin
  3304. if StatusOK then
  3305. with FLoadedImage.FileHeader^ do
  3306. case Index of
  3307. JclPeHeader_Signature:
  3308. Result := IntToHex(Signature, 8);
  3309. JclPeHeader_Machine:
  3310. Result := GetMachineString(FileHeader.Machine);
  3311. JclPeHeader_NumberOfSections:
  3312. Result := IntToHex(FileHeader.NumberOfSections, 4);
  3313. JclPeHeader_TimeDateStamp:
  3314. Result := IntToHex(FileHeader.TimeDateStamp, 8);
  3315. JclPeHeader_PointerToSymbolTable:
  3316. Result := IntToHex(FileHeader.PointerToSymbolTable, 8);
  3317. JclPeHeader_NumberOfSymbols:
  3318. Result := IntToHex(FileHeader.NumberOfSymbols, 8);
  3319. JclPeHeader_SizeOfOptionalHeader:
  3320. Result := IntToHex(FileHeader.SizeOfOptionalHeader, 4);
  3321. JclPeHeader_Characteristics:
  3322. Result := IntToHex(FileHeader.Characteristics, 4);
  3323. JclPeHeader_Magic..JclPeHeader_NumberOfRvaAndSizes:
  3324. case Target of
  3325. taWin32:
  3326. Result := GetHeaderValues32(Index);
  3327. taWin64:
  3328. Result := GetHeaderValues64(Index);
  3329. //taUnknown:
  3330. else
  3331. Result := '';
  3332. end;
  3333. else
  3334. Result := '';
  3335. end
  3336. else
  3337. Result := '';
  3338. end;
  3339. function TJclPeImage.GetImageSectionCount: Integer;
  3340. begin
  3341. Result := FImageSections.Count;
  3342. end;
  3343. function TJclPeImage.GetImageSectionFullNames(Index: Integer): string;
  3344. var
  3345. Offset: Integer;
  3346. begin
  3347. Result := ImageSectionNames[Index];
  3348. if (Length(Result) > 0) and (Result[1] = '/') and TryStrToInt(Copy(Result, 2, MaxInt), Offset) then
  3349. Result := GetNameInStringTable(Offset);
  3350. end;
  3351. function TJclPeImage.GetImageSectionHeaders(Index: Integer): TImageSectionHeader;
  3352. begin
  3353. Result := PImageSectionHeader(FImageSections.Objects[Index])^;
  3354. end;
  3355. function TJclPeImage.GetImageSectionNameFromRva(const Rva: DWORD): string;
  3356. begin
  3357. Result := GetSectionName(RvaToSection(Rva));
  3358. end;
  3359. function TJclPeImage.GetImageSectionNames(Index: Integer): string;
  3360. begin
  3361. Result := FImageSections[Index];
  3362. end;
  3363. function TJclPeImage.GetImportList: TJclPeImportList;
  3364. begin
  3365. if FImportList = nil then
  3366. FImportList := TJclPeImportList.Create(Self);
  3367. Result := FImportList;
  3368. end;
  3369. function TJclPeImage.GetLoadConfigValues(Index: TJclLoadConfig): string;
  3370. function GetLoadConfigValues32(Index: TJclLoadConfig): string;
  3371. var
  3372. LoadConfig: PIMAGE_LOAD_CONFIG_DIRECTORY32;
  3373. begin
  3374. LoadConfig := DirectoryEntryToData(IMAGE_DIRECTORY_ENTRY_LOAD_CONFIG);
  3375. if LoadConfig <> nil then
  3376. with LoadConfig^ do
  3377. case Index of
  3378. JclLoadConfig_Characteristics:
  3379. Result := IntToHex(Size, 8);
  3380. JclLoadConfig_TimeDateStamp:
  3381. Result := IntToHex(TimeDateStamp, 8);
  3382. JclLoadConfig_Version:
  3383. Result := FormatVersionString(MajorVersion, MinorVersion);
  3384. JclLoadConfig_GlobalFlagsClear:
  3385. Result := IntToHex(GlobalFlagsClear, 8);
  3386. JclLoadConfig_GlobalFlagsSet:
  3387. Result := IntToHex(GlobalFlagsSet, 8);
  3388. JclLoadConfig_CriticalSectionDefaultTimeout:
  3389. Result := IntToHex(CriticalSectionDefaultTimeout, 8);
  3390. JclLoadConfig_DeCommitFreeBlockThreshold:
  3391. Result := IntToHex(DeCommitFreeBlockThreshold, 8);
  3392. JclLoadConfig_DeCommitTotalFreeThreshold:
  3393. Result := IntToHex(DeCommitTotalFreeThreshold, 8);
  3394. JclLoadConfig_LockPrefixTable:
  3395. Result := IntToHex(LockPrefixTable, 8);
  3396. JclLoadConfig_MaximumAllocationSize:
  3397. Result := IntToHex(MaximumAllocationSize, 8);
  3398. JclLoadConfig_VirtualMemoryThreshold:
  3399. Result := IntToHex(VirtualMemoryThreshold, 8);
  3400. JclLoadConfig_ProcessHeapFlags:
  3401. Result := IntToHex(ProcessHeapFlags, 8);
  3402. JclLoadConfig_ProcessAffinityMask:
  3403. Result := IntToHex(ProcessAffinityMask, 8);
  3404. JclLoadConfig_CSDVersion:
  3405. Result := IntToHex(CSDVersion, 4);
  3406. JclLoadConfig_Reserved1:
  3407. Result := IntToHex(Reserved1, 4);
  3408. JclLoadConfig_EditList:
  3409. Result := IntToHex(EditList, 8);
  3410. JclLoadConfig_Reserved:
  3411. Result := LoadResString(@RsPeReserved);
  3412. end;
  3413. end;
  3414. function GetLoadConfigValues64(Index: TJclLoadConfig): string;
  3415. var
  3416. LoadConfig: PIMAGE_LOAD_CONFIG_DIRECTORY64;
  3417. begin
  3418. LoadConfig := DirectoryEntryToData(IMAGE_DIRECTORY_ENTRY_LOAD_CONFIG);
  3419. if LoadConfig <> nil then
  3420. with LoadConfig^ do
  3421. case Index of
  3422. JclLoadConfig_Characteristics:
  3423. Result := IntToHex(Size, 8);
  3424. JclLoadConfig_TimeDateStamp:
  3425. Result := IntToHex(TimeDateStamp, 8);
  3426. JclLoadConfig_Version:
  3427. Result := FormatVersionString(MajorVersion, MinorVersion);
  3428. JclLoadConfig_GlobalFlagsClear:
  3429. Result := IntToHex(GlobalFlagsClear, 8);
  3430. JclLoadConfig_GlobalFlagsSet:
  3431. Result := IntToHex(GlobalFlagsSet, 8);
  3432. JclLoadConfig_CriticalSectionDefaultTimeout:
  3433. Result := IntToHex(CriticalSectionDefaultTimeout, 8);
  3434. JclLoadConfig_DeCommitFreeBlockThreshold:
  3435. Result := IntToHex(DeCommitFreeBlockThreshold, 16);
  3436. JclLoadConfig_DeCommitTotalFreeThreshold:
  3437. Result := IntToHex(DeCommitTotalFreeThreshold, 16);
  3438. JclLoadConfig_LockPrefixTable:
  3439. Result := IntToHex(LockPrefixTable, 16);
  3440. JclLoadConfig_MaximumAllocationSize:
  3441. Result := IntToHex(MaximumAllocationSize, 16);
  3442. JclLoadConfig_VirtualMemoryThreshold:
  3443. Result := IntToHex(VirtualMemoryThreshold, 16);
  3444. JclLoadConfig_ProcessHeapFlags:
  3445. Result := IntToHex(ProcessHeapFlags, 8);
  3446. JclLoadConfig_ProcessAffinityMask:
  3447. Result := IntToHex(ProcessAffinityMask, 16);
  3448. JclLoadConfig_CSDVersion:
  3449. Result := IntToHex(CSDVersion, 4);
  3450. JclLoadConfig_Reserved1:
  3451. Result := IntToHex(Reserved1, 4);
  3452. JclLoadConfig_EditList:
  3453. Result := IntToHex(EditList, 16);
  3454. JclLoadConfig_Reserved:
  3455. Result := LoadResString(@RsPeReserved);
  3456. end;
  3457. end;
  3458. begin
  3459. Result := '';
  3460. case Target of
  3461. taWin32:
  3462. Result := GetLoadConfigValues32(Index);
  3463. taWin64:
  3464. Result := GetLoadConfigValues64(Index);
  3465. end;
  3466. end;
  3467. function TJclPeImage.GetMappedAddress: TJclAddr;
  3468. begin
  3469. if StatusOK then
  3470. Result := TJclAddr(LoadedImage.MappedAddress)
  3471. else
  3472. Result := 0;
  3473. end;
  3474. function TJclPeImage.GetNameInStringTable(Offset: ULONG): string;
  3475. var
  3476. Index: Integer;
  3477. begin
  3478. Dec(Offset, SizeOf(ULONG));
  3479. Index := 0;
  3480. while (Offset > 0) and (Index < FStringTable.Count) do
  3481. begin
  3482. Dec(Offset, Length(FStringTable[Index]) + 1);
  3483. if Offset > 0 then
  3484. Inc(Index);
  3485. end;
  3486. if Offset = 0 then
  3487. Result := FStringTable[Index]
  3488. else
  3489. Result := '';
  3490. end;
  3491. function TJclPeImage.GetOptionalHeader32: TImageOptionalHeader32;
  3492. begin
  3493. if Target = taWin32 then
  3494. Result := PImageNtHeaders32(FLoadedImage.FileHeader)^.OptionalHeader
  3495. else
  3496. ZeroMemory(@Result, SizeOf(Result));
  3497. end;
  3498. function TJclPeImage.GetOptionalHeader64: TImageOptionalHeader64;
  3499. begin
  3500. if Target = taWin64 then
  3501. Result := PImageNtHeaders64(FLoadedImage.FileHeader)^.OptionalHeader
  3502. else
  3503. ZeroMemory(@Result, SizeOf(Result));
  3504. end;
  3505. function TJclPeImage.GetRelocationList: TJclPeRelocList;
  3506. begin
  3507. if FRelocationList = nil then
  3508. FRelocationList := TJclPeRelocList.Create(Self);
  3509. Result := FRelocationList;
  3510. end;
  3511. function TJclPeImage.GetResourceList: TJclPeRootResourceList;
  3512. begin
  3513. if FResourceList = nil then
  3514. begin
  3515. FResourceVA := Directories[IMAGE_DIRECTORY_ENTRY_RESOURCE].VirtualAddress;
  3516. if FResourceVA <> 0 then
  3517. FResourceVA := TJclAddr(RvaToVa(FResourceVA));
  3518. FResourceList := TJclPeRootResourceList.Create(Self, nil, PImageResourceDirectory(FResourceVA));
  3519. end;
  3520. Result := FResourceList;
  3521. end;
  3522. function TJclPeImage.GetSectionHeader(const SectionName: string;
  3523. out Header: PImageSectionHeader): Boolean;
  3524. var
  3525. I: Integer;
  3526. begin
  3527. I := FImageSections.IndexOf(SectionName);
  3528. if I = -1 then
  3529. begin
  3530. Header := nil;
  3531. Result := False;
  3532. end
  3533. else
  3534. begin
  3535. Header := PImageSectionHeader(FImageSections.Objects[I]);
  3536. Result := True;
  3537. end;
  3538. end;
  3539. function TJclPeImage.GetSectionName(Header: PImageSectionHeader): string;
  3540. var
  3541. I: Integer;
  3542. begin
  3543. I := FImageSections.IndexOfObject(TObject(Header));
  3544. if I = -1 then
  3545. Result := ''
  3546. else
  3547. Result := FImageSections[I];
  3548. end;
  3549. function TJclPeImage.GetStringTableCount: Integer;
  3550. begin
  3551. Result := FStringTable.Count;
  3552. end;
  3553. function TJclPeImage.GetStringTableItem(Index: Integer): string;
  3554. begin
  3555. Result := FStringTable[Index];
  3556. end;
  3557. function TJclPeImage.GetUnusedHeaderBytes: TImageDataDirectory;
  3558. begin
  3559. CheckNotAttached;
  3560. Result.Size := 0;
  3561. Result.VirtualAddress := GetImageUnusedHeaderBytes(FLoadedImage, Result.Size);
  3562. if Result.VirtualAddress = 0 then
  3563. RaiseLastOSError;
  3564. end;
  3565. function TJclPeImage.GetVersionInfo: TJclFileVersionInfo;
  3566. var
  3567. VersionInfoResource: TJclPeResourceItem;
  3568. begin
  3569. if (FVersionInfo = nil) and VersionInfoAvailable then
  3570. begin
  3571. VersionInfoResource := ResourceList.FindResource(rtVersion, '1').List[0];
  3572. with VersionInfoResource do
  3573. try
  3574. FVersionInfo := TJclFileVersionInfo.Attach(RawEntryData, RawEntryDataSize);
  3575. except
  3576. FreeAndNil(FVersionInfo);
  3577. end;
  3578. end;
  3579. Result := FVersionInfo;
  3580. end;
  3581. function TJclPeImage.GetVersionInfoAvailable: Boolean;
  3582. begin
  3583. Result := StatusOK and (ResourceList.FindResource(rtVersion, '1') <> nil);
  3584. end;
  3585. class function TJclPeImage.HeaderNames(Index: TJclPeHeader): string;
  3586. begin
  3587. case Index of
  3588. JclPeHeader_Signature:
  3589. Result := LoadResString(@RsPeSignature);
  3590. JclPeHeader_Machine:
  3591. Result := LoadResString(@RsPeMachine);
  3592. JclPeHeader_NumberOfSections:
  3593. Result := LoadResString(@RsPeNumberOfSections);
  3594. JclPeHeader_TimeDateStamp:
  3595. Result := LoadResString(@RsPeTimeDateStamp);
  3596. JclPeHeader_PointerToSymbolTable:
  3597. Result := LoadResString(@RsPePointerToSymbolTable);
  3598. JclPeHeader_NumberOfSymbols:
  3599. Result := LoadResString(@RsPeNumberOfSymbols);
  3600. JclPeHeader_SizeOfOptionalHeader:
  3601. Result := LoadResString(@RsPeSizeOfOptionalHeader);
  3602. JclPeHeader_Characteristics:
  3603. Result := LoadResString(@RsPeCharacteristics);
  3604. JclPeHeader_Magic:
  3605. Result := LoadResString(@RsPeMagic);
  3606. JclPeHeader_LinkerVersion:
  3607. Result := LoadResString(@RsPeLinkerVersion);
  3608. JclPeHeader_SizeOfCode:
  3609. Result := LoadResString(@RsPeSizeOfCode);
  3610. JclPeHeader_SizeOfInitializedData:
  3611. Result := LoadResString(@RsPeSizeOfInitializedData);
  3612. JclPeHeader_SizeOfUninitializedData:
  3613. Result := LoadResString(@RsPeSizeOfUninitializedData);
  3614. JclPeHeader_AddressOfEntryPoint:
  3615. Result := LoadResString(@RsPeAddressOfEntryPoint);
  3616. JclPeHeader_BaseOfCode:
  3617. Result := LoadResString(@RsPeBaseOfCode);
  3618. JclPeHeader_BaseOfData:
  3619. Result := LoadResString(@RsPeBaseOfData);
  3620. JclPeHeader_ImageBase:
  3621. Result := LoadResString(@RsPeImageBase);
  3622. JclPeHeader_SectionAlignment:
  3623. Result := LoadResString(@RsPeSectionAlignment);
  3624. JclPeHeader_FileAlignment:
  3625. Result := LoadResString(@RsPeFileAlignment);
  3626. JclPeHeader_OperatingSystemVersion:
  3627. Result := LoadResString(@RsPeOperatingSystemVersion);
  3628. JclPeHeader_ImageVersion:
  3629. Result := LoadResString(@RsPeImageVersion);
  3630. JclPeHeader_SubsystemVersion:
  3631. Result := LoadResString(@RsPeSubsystemVersion);
  3632. JclPeHeader_Win32VersionValue:
  3633. Result := LoadResString(@RsPeWin32VersionValue);
  3634. JclPeHeader_SizeOfImage:
  3635. Result := LoadResString(@RsPeSizeOfImage);
  3636. JclPeHeader_SizeOfHeaders:
  3637. Result := LoadResString(@RsPeSizeOfHeaders);
  3638. JclPeHeader_CheckSum:
  3639. Result := LoadResString(@RsPeCheckSum);
  3640. JclPeHeader_Subsystem:
  3641. Result := LoadResString(@RsPeSubsystem);
  3642. JclPeHeader_DllCharacteristics:
  3643. Result := LoadResString(@RsPeDllCharacteristics);
  3644. JclPeHeader_SizeOfStackReserve:
  3645. Result := LoadResString(@RsPeSizeOfStackReserve);
  3646. JclPeHeader_SizeOfStackCommit:
  3647. Result := LoadResString(@RsPeSizeOfStackCommit);
  3648. JclPeHeader_SizeOfHeapReserve:
  3649. Result := LoadResString(@RsPeSizeOfHeapReserve);
  3650. JclPeHeader_SizeOfHeapCommit:
  3651. Result := LoadResString(@RsPeSizeOfHeapCommit);
  3652. JclPeHeader_LoaderFlags:
  3653. Result := LoadResString(@RsPeLoaderFlags);
  3654. JclPeHeader_NumberOfRvaAndSizes:
  3655. Result := LoadResString(@RsPeNumberOfRvaAndSizes);
  3656. else
  3657. Result := '';
  3658. end;
  3659. end;
  3660. function TJclPeImage.IsBrokenFormat: Boolean;
  3661. function IsBrokenFormat32: Boolean;
  3662. var
  3663. OptionalHeader: TImageOptionalHeader32;
  3664. begin
  3665. OptionalHeader := OptionalHeader32;
  3666. Result := not ((OptionalHeader.AddressOfEntryPoint = 0) or IsCLR);
  3667. if Result then
  3668. begin
  3669. Result := (ImageSectionCount = 0);
  3670. if not Result then
  3671. with ImageSectionHeaders[0] do
  3672. Result := (VirtualAddress <> OptionalHeader.BaseOfCode) or (SizeOfRawData = 0) or
  3673. (OptionalHeader.AddressOfEntryPoint > VirtualAddress + Misc.VirtualSize) or
  3674. (Characteristics and (IMAGE_SCN_CNT_CODE or IMAGE_SCN_MEM_WRITE) <> IMAGE_SCN_CNT_CODE);
  3675. end;
  3676. end;
  3677. function IsBrokenFormat64: Boolean;
  3678. var
  3679. OptionalHeader: TImageOptionalHeader64;
  3680. begin
  3681. OptionalHeader := OptionalHeader64;
  3682. Result := not ((OptionalHeader.AddressOfEntryPoint = 0) or IsCLR);
  3683. if Result then
  3684. begin
  3685. Result := (ImageSectionCount = 0);
  3686. if not Result then
  3687. with ImageSectionHeaders[0] do
  3688. Result := (VirtualAddress <> OptionalHeader.BaseOfCode) or (SizeOfRawData = 0) or
  3689. (OptionalHeader.AddressOfEntryPoint > VirtualAddress + Misc.VirtualSize) or
  3690. (Characteristics and (IMAGE_SCN_CNT_CODE or IMAGE_SCN_MEM_WRITE) <> IMAGE_SCN_CNT_CODE);
  3691. end;
  3692. end;
  3693. begin
  3694. case Target of
  3695. taWin32:
  3696. Result := IsBrokenFormat32;
  3697. taWin64:
  3698. Result := IsBrokenFormat64;
  3699. //taUnknown:
  3700. else
  3701. Result := False; // don't know how to check it
  3702. end;
  3703. end;
  3704. function TJclPeImage.IsCLR: Boolean;
  3705. begin
  3706. Result := DirectoryExists[IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR] and CLRHeader.HasMetadata;
  3707. end;
  3708. function TJclPeImage.IsSystemImage: Boolean;
  3709. begin
  3710. Result := StatusOK and FLoadedImage.fSystemImage;
  3711. end;
  3712. class function TJclPeImage.LoadConfigNames(Index: TJclLoadConfig): string;
  3713. begin
  3714. case Index of
  3715. JclLoadConfig_Characteristics:
  3716. Result := LoadResString(@RsPeCharacteristics);
  3717. JclLoadConfig_TimeDateStamp:
  3718. Result := LoadResString(@RsPeTimeDateStamp);
  3719. JclLoadConfig_Version:
  3720. Result := LoadResString(@RsPeVersion);
  3721. JclLoadConfig_GlobalFlagsClear:
  3722. Result := LoadResString(@RsPeGlobalFlagsClear);
  3723. JclLoadConfig_GlobalFlagsSet:
  3724. Result := LoadResString(@RsPeGlobalFlagsSet);
  3725. JclLoadConfig_CriticalSectionDefaultTimeout:
  3726. Result := LoadResString(@RsPeCriticalSectionDefaultTimeout);
  3727. JclLoadConfig_DeCommitFreeBlockThreshold:
  3728. Result := LoadResString(@RsPeDeCommitFreeBlockThreshold);
  3729. JclLoadConfig_DeCommitTotalFreeThreshold:
  3730. Result := LoadResString(@RsPeDeCommitTotalFreeThreshold);
  3731. JclLoadConfig_LockPrefixTable:
  3732. Result := LoadResString(@RsPeLockPrefixTable);
  3733. JclLoadConfig_MaximumAllocationSize:
  3734. Result := LoadResString(@RsPeMaximumAllocationSize);
  3735. JclLoadConfig_VirtualMemoryThreshold:
  3736. Result := LoadResString(@RsPeVirtualMemoryThreshold);
  3737. JclLoadConfig_ProcessHeapFlags:
  3738. Result := LoadResString(@RsPeProcessHeapFlags);
  3739. JclLoadConfig_ProcessAffinityMask:
  3740. Result := LoadResString(@RsPeProcessAffinityMask);
  3741. JclLoadConfig_CSDVersion:
  3742. Result := LoadResString(@RsPeCSDVersion);
  3743. JclLoadConfig_Reserved1:
  3744. Result := LoadResString(@RsPeReserved);
  3745. JclLoadConfig_EditList:
  3746. Result := LoadResString(@RsPeEditList);
  3747. JclLoadConfig_Reserved:
  3748. Result := LoadResString(@RsPeReserved);
  3749. else
  3750. Result := '';
  3751. end;
  3752. end;
  3753. procedure TJclPeImage.RaiseStatusException;
  3754. begin
  3755. if not FNoExceptions then
  3756. case FStatus of
  3757. stNotPE:
  3758. raise EJclPeImageError.CreateRes(@RsPeNotPE);
  3759. stNotFound:
  3760. raise EJclPeImageError.CreateResFmt(@RsPeCantOpen, [FFileName]);
  3761. stNotSupported:
  3762. raise EJclPeImageError.CreateRes(@RsPeUnknownTarget);
  3763. stError:
  3764. RaiseLastOSError;
  3765. end;
  3766. end;
  3767. function TJclPeImage.RawToVa(Raw: DWORD): Pointer;
  3768. begin
  3769. Result := Pointer(TJclAddr(FLoadedImage.MappedAddress) + Raw);
  3770. end;
  3771. procedure TJclPeImage.ReadImageSections;
  3772. var
  3773. I: Integer;
  3774. Header: PImageSectionHeader;
  3775. UTF8Name: TUTF8String;
  3776. SectionName: string;
  3777. begin
  3778. if not StatusOK then
  3779. Exit;
  3780. Header := FLoadedImage.Sections;
  3781. for I := 0 to FLoadedImage.NumberOfSections - 1 do
  3782. begin
  3783. SetLength(UTF8Name, IMAGE_SIZEOF_SHORT_NAME);
  3784. Move(Header.Name[0], UTF8Name[1], IMAGE_SIZEOF_SHORT_NAME * SizeOf(AnsiChar));
  3785. StrResetLength(UTF8Name);
  3786. if not TryUTF8ToString(UTF8Name, SectionName) then
  3787. SectionName := string(UTF8Name);
  3788. FImageSections.AddObject(SectionName, Pointer(Header));
  3789. Inc(Header);
  3790. end;
  3791. end;
  3792. procedure TJclPeImage.ReadStringTable;
  3793. var
  3794. SymbolTable: DWORD;
  3795. StringTablePtr: PAnsiChar;
  3796. Ptr: PAnsiChar;
  3797. ByteSize: ULONG;
  3798. Start: PAnsiChar;
  3799. StringEntry: AnsiString;
  3800. begin
  3801. SymbolTable := LoadedImage.FileHeader.FileHeader.PointerToSymbolTable;
  3802. if SymbolTable = 0 then
  3803. Exit;
  3804. StringTablePtr := PAnsiChar(LoadedImage.MappedAddress) +
  3805. SymbolTable +
  3806. (LoadedImage.FileHeader.FileHeader.NumberOfSymbols * SizeOf(IMAGE_SYMBOL));
  3807. ByteSize := PULONG(StringTablePtr)^;
  3808. Ptr := StringTablePtr + SizeOf(ByteSize);
  3809. while Ptr < StringTablePtr + ByteSize do
  3810. begin
  3811. Start := Ptr;
  3812. while (Ptr^ <> #0) and (Ptr < StringTablePtr + ByteSize) do
  3813. Inc(Ptr);
  3814. if Start <> Ptr then
  3815. begin
  3816. SetLength(StringEntry, Ptr - Start);
  3817. Move(Start^, StringEntry[1], Ptr - Start);
  3818. FStringTable.Add(string(StringEntry));
  3819. end;
  3820. Inc(Ptr); // to skip the #0 character
  3821. end;
  3822. end;
  3823. function TJclPeImage.ResourceItemCreate(AEntry: PImageResourceDirectoryEntry;
  3824. AParentItem: TJclPeResourceItem): TJclPeResourceItem;
  3825. begin
  3826. Result := TJclPeResourceItem.Create(Self, AParentItem, AEntry);
  3827. end;
  3828. function TJclPeImage.ResourceListCreate(ADirectory: PImageResourceDirectory;
  3829. AParentItem: TJclPeResourceItem): TJclPeResourceList;
  3830. begin
  3831. Result := TJclPeResourceList.Create(Self, AParentItem, ADirectory);
  3832. end;
  3833. function TJclPeImage.RvaToSection(Rva: DWORD): PImageSectionHeader;
  3834. var
  3835. I: Integer;
  3836. SectionHeader: PImageSectionHeader;
  3837. EndRVA: DWORD;
  3838. begin
  3839. Result := ImageRvaToSection(FLoadedImage.FileHeader, FLoadedImage.MappedAddress, Rva);
  3840. if Result = nil then
  3841. for I := 0 to FImageSections.Count - 1 do
  3842. begin
  3843. SectionHeader := PImageSectionHeader(FImageSections.Objects[I]);
  3844. if SectionHeader^.SizeOfRawData = 0 then
  3845. EndRVA := SectionHeader^.Misc.VirtualSize
  3846. else
  3847. EndRVA := SectionHeader^.SizeOfRawData;
  3848. Inc(EndRVA, SectionHeader^.VirtualAddress);
  3849. if (SectionHeader^.VirtualAddress <= Rva) and (EndRVA >= Rva) then
  3850. begin
  3851. Result := SectionHeader;
  3852. Break;
  3853. end;
  3854. end;
  3855. end;
  3856. function TJclPeImage.RvaToVa(Rva: DWORD): Pointer;
  3857. begin
  3858. if FAttachedImage then
  3859. Result := Pointer(TJclAddr(FLoadedImage.MappedAddress) + Rva)
  3860. else
  3861. Result := ImageRvaToVa(FLoadedImage.FileHeader, FLoadedImage.MappedAddress, Rva, nil);
  3862. end;
  3863. function TJclPeImage.RvaToVaEx(Rva: DWORD): Pointer;
  3864. function RvaToVaEx32(Rva: DWORD): Pointer;
  3865. var
  3866. OptionalHeader: TImageOptionalHeader32;
  3867. begin
  3868. OptionalHeader := OptionalHeader32;
  3869. if (Rva >= OptionalHeader.ImageBase) and (Rva < (OptionalHeader.ImageBase + FLoadedImage.SizeOfImage)) then
  3870. Dec(Rva, OptionalHeader.ImageBase);
  3871. Result := RvaToVa(Rva);
  3872. end;
  3873. function RvaToVaEx64(Rva: DWORD): Pointer;
  3874. var
  3875. OptionalHeader: TImageOptionalHeader64;
  3876. begin
  3877. OptionalHeader := OptionalHeader64;
  3878. if (Rva >= OptionalHeader.ImageBase) and (Rva < (OptionalHeader.ImageBase + FLoadedImage.SizeOfImage)) then
  3879. Dec(Rva, OptionalHeader.ImageBase);
  3880. Result := RvaToVa(Rva);
  3881. end;
  3882. begin
  3883. case Target of
  3884. taWin32:
  3885. Result := RvaToVaEx32(Rva);
  3886. taWin64:
  3887. Result := RvaToVaEx64(Rva);
  3888. //taUnknown:
  3889. else
  3890. Result := nil;
  3891. end;
  3892. end;
  3893. procedure TJclPeImage.SetFileName(const Value: TFileName);
  3894. begin
  3895. if FFileName <> Value then
  3896. begin
  3897. Clear;
  3898. FFileName := Value;
  3899. if FFileName = '' then
  3900. Exit;
  3901. // OF: possible loss of data
  3902. if MapAndLoad(PAnsiChar(AnsiString(FFileName)), nil, FLoadedImage, True, FReadOnlyAccess) then
  3903. begin
  3904. FTarget := PeMapImgTarget(FLoadedImage.MappedAddress);
  3905. if FTarget <> taUnknown then
  3906. begin
  3907. FStatus := stOk;
  3908. ReadImageSections;
  3909. ReadStringTable;
  3910. AfterOpen;
  3911. end
  3912. else
  3913. FStatus := stNotSupported;
  3914. end
  3915. else
  3916. case GetLastError of
  3917. ERROR_SUCCESS:
  3918. FStatus := stNotPE;
  3919. ERROR_FILE_NOT_FOUND:
  3920. FStatus := stNotFound;
  3921. else
  3922. FStatus := stError;
  3923. end;
  3924. RaiseStatusException;
  3925. end;
  3926. end;
  3927. class function TJclPeImage.ShortSectionInfo(Characteristics: DWORD): string;
  3928. type
  3929. TSectionCharacteristics = packed record
  3930. Mask: DWORD;
  3931. InfoChar: Char;
  3932. end;
  3933. const
  3934. Info: array [1..8] of TSectionCharacteristics = (
  3935. (Mask: IMAGE_SCN_CNT_CODE; InfoChar: 'C'),
  3936. (Mask: IMAGE_SCN_MEM_EXECUTE; InfoChar: 'E'),
  3937. (Mask: IMAGE_SCN_MEM_READ; InfoChar: 'R'),
  3938. (Mask: IMAGE_SCN_MEM_WRITE; InfoChar: 'W'),
  3939. (Mask: IMAGE_SCN_CNT_INITIALIZED_DATA; InfoChar: 'I'),
  3940. (Mask: IMAGE_SCN_CNT_UNINITIALIZED_DATA; InfoChar: 'U'),
  3941. (Mask: IMAGE_SCN_MEM_SHARED; InfoChar: 'S'),
  3942. (Mask: IMAGE_SCN_MEM_DISCARDABLE; InfoChar: 'D')
  3943. );
  3944. var
  3945. I: Integer;
  3946. begin
  3947. SetLength(Result, High(Info));
  3948. Result := '';
  3949. for I := Low(Info) to High(Info) do
  3950. with Info[I] do
  3951. if (Characteristics and Mask) = Mask then
  3952. Result := Result + InfoChar;
  3953. end;
  3954. function TJclPeImage.StatusOK: Boolean;
  3955. begin
  3956. Result := (FStatus = stOk);
  3957. end;
  3958. class function TJclPeImage.StampToDateTime(TimeDateStamp: DWORD): TDateTime;
  3959. begin
  3960. Result := TimeDateStamp / SecsPerDay + UnixTimeStart
  3961. end;
  3962. procedure TJclPeImage.TryGetNamesForOrdinalImports;
  3963. begin
  3964. if StatusOK then
  3965. begin
  3966. GetImportList;
  3967. FImportList.TryGetNamesForOrdinalImports;
  3968. end;
  3969. end;
  3970. function TJclPeImage.VerifyCheckSum: Boolean;
  3971. function VerifyCheckSum32: Boolean;
  3972. var
  3973. OptionalHeader: TImageOptionalHeader32;
  3974. begin
  3975. OptionalHeader := OptionalHeader32;
  3976. Result := StatusOK and ((OptionalHeader.CheckSum = 0) or (CalculateCheckSum = OptionalHeader.CheckSum));
  3977. end;
  3978. function VerifyCheckSum64: Boolean;
  3979. var
  3980. OptionalHeader: TImageOptionalHeader64;
  3981. begin
  3982. OptionalHeader := OptionalHeader64;
  3983. Result := StatusOK and ((OptionalHeader.CheckSum = 0) or (CalculateCheckSum = OptionalHeader.CheckSum));
  3984. end;
  3985. begin
  3986. CheckNotAttached;
  3987. case Target of
  3988. taWin32:
  3989. Result := VerifyCheckSum32;
  3990. taWin64:
  3991. Result := VerifyCheckSum64;
  3992. //taUnknown: ;
  3993. else
  3994. Result := True;
  3995. end;
  3996. end;
  3997. {$IFDEF BORLAND}
  3998. //=== { TJclPeBorImagesCache } ===============================================
  3999. function TJclPeBorImagesCache.GetImages(const FileName: TFileName): TJclPeBorImage;
  4000. begin
  4001. Result := TJclPeBorImage(inherited Images[FileName]);
  4002. end;
  4003. function TJclPeBorImagesCache.GetPeImageClass: TJclPeImageClass;
  4004. begin
  4005. Result := TJclPeBorImage;
  4006. end;
  4007. //=== { TJclPePackageInfo } ==================================================
  4008. constructor TJclPePackageInfo.Create(ALibHandle: THandle);
  4009. begin
  4010. FContains := TStringList.Create;
  4011. FRequires := TStringList.Create;
  4012. FEnsureExtension := True;
  4013. FSorted := True;
  4014. ReadPackageInfo(ALibHandle);
  4015. end;
  4016. destructor TJclPePackageInfo.Destroy;
  4017. begin
  4018. FreeAndNil(FContains);
  4019. FreeAndNil(FRequires);
  4020. inherited Destroy;
  4021. end;
  4022. function TJclPePackageInfo.GetContains: TStrings;
  4023. begin
  4024. Result := FContains;
  4025. end;
  4026. function TJclPePackageInfo.GetContainsCount: Integer;
  4027. begin
  4028. Result := Contains.Count;
  4029. end;
  4030. function TJclPePackageInfo.GetContainsFlags(Index: Integer): Byte;
  4031. begin
  4032. Result := Byte(Contains.Objects[Index]);
  4033. end;
  4034. function TJclPePackageInfo.GetContainsNames(Index: Integer): string;
  4035. begin
  4036. Result := Contains[Index];
  4037. end;
  4038. function TJclPePackageInfo.GetRequires: TStrings;
  4039. begin
  4040. Result := FRequires;
  4041. end;
  4042. function TJclPePackageInfo.GetRequiresCount: Integer;
  4043. begin
  4044. Result := Requires.Count;
  4045. end;
  4046. function TJclPePackageInfo.GetRequiresNames(Index: Integer): string;
  4047. begin
  4048. Result := Requires[Index];
  4049. if FEnsureExtension then
  4050. StrEnsureSuffix(BinaryExtensionPackage, Result);
  4051. end;
  4052. class function TJclPePackageInfo.PackageModuleTypeToString(Flags: Cardinal): string;
  4053. begin
  4054. case Flags and pfModuleTypeMask of
  4055. pfExeModule, pfModuleTypeMask:
  4056. Result := LoadResString(@RsPePkgExecutable);
  4057. pfPackageModule:
  4058. Result := LoadResString(@RsPePkgPackage);
  4059. pfLibraryModule:
  4060. Result := LoadResString(@PsPePkgLibrary);
  4061. else
  4062. Result := '';
  4063. end;
  4064. end;
  4065. class function TJclPePackageInfo.PackageOptionsToString(Flags: Cardinal): string;
  4066. begin
  4067. Result := '';
  4068. AddFlagTextRes(Result, @RsPePkgNeverBuild, Flags, pfNeverBuild);
  4069. AddFlagTextRes(Result, @RsPePkgDesignOnly, Flags, pfDesignOnly);
  4070. AddFlagTextRes(Result, @RsPePkgRunOnly, Flags, pfRunOnly);
  4071. AddFlagTextRes(Result, @RsPePkgIgnoreDupUnits, Flags, pfIgnoreDupUnits);
  4072. end;
  4073. class function TJclPePackageInfo.ProducerToString(Flags: Cardinal): string;
  4074. begin
  4075. case Flags and pfProducerMask of
  4076. pfV3Produced:
  4077. Result := LoadResString(@RsPePkgV3Produced);
  4078. pfProducerUndefined:
  4079. Result := LoadResString(@RsPePkgProducerUndefined);
  4080. pfBCB4Produced:
  4081. Result := LoadResString(@RsPePkgBCB4Produced);
  4082. pfDelphi4Produced:
  4083. Result := LoadResString(@RsPePkgDelphi4Produced);
  4084. else
  4085. Result := '';
  4086. end;
  4087. end;
  4088. procedure PackageInfoProc(const Name: string; NameType: TNameType; AFlags: Byte; Param: Pointer);
  4089. begin
  4090. with TJclPePackageInfo(Param) do
  4091. case NameType of
  4092. ntContainsUnit:
  4093. Contains.AddObject(Name, Pointer(AFlags));
  4094. ntRequiresPackage:
  4095. Requires.Add(Name);
  4096. ntDcpBpiName:
  4097. SetDcpName(Name);
  4098. end;
  4099. end;
  4100. procedure TJclPePackageInfo.ReadPackageInfo(ALibHandle: THandle);
  4101. var
  4102. DescrResInfo: HRSRC;
  4103. DescrResData: HGLOBAL;
  4104. begin
  4105. FAvailable := FindResource(ALibHandle, PackageInfoResName, RT_RCDATA) <> 0;
  4106. if FAvailable then
  4107. begin
  4108. GetPackageInfo(ALibHandle, Self, FFlags, PackageInfoProc);
  4109. if FDcpName = '' then
  4110. FDcpName := PathExtractFileNameNoExt(GetModulePath(ALibHandle)) + CompilerExtensionDCP;
  4111. if FSorted then
  4112. begin
  4113. FContains.Sort;
  4114. FRequires.Sort;
  4115. end;
  4116. end;
  4117. DescrResInfo := FindResource(ALibHandle, DescriptionResName, RT_RCDATA);
  4118. if DescrResInfo <> 0 then
  4119. begin
  4120. DescrResData := LoadResource(ALibHandle, DescrResInfo);
  4121. if DescrResData <> 0 then
  4122. begin
  4123. FDescription := WideCharLenToString(LockResource(DescrResData),
  4124. SizeofResource(ALibHandle, DescrResInfo));
  4125. StrResetLength(FDescription);
  4126. end;
  4127. end;
  4128. end;
  4129. procedure TJclPePackageInfo.SetDcpName(const Value: string);
  4130. begin
  4131. FDcpName := Value;
  4132. end;
  4133. class function TJclPePackageInfo.UnitInfoFlagsToString(UnitFlags: Byte): string;
  4134. begin
  4135. Result := '';
  4136. AddFlagTextRes(Result, @RsPePkgMain, UnitFlags, ufMainUnit);
  4137. AddFlagTextRes(Result, @RsPePkgPackage, UnitFlags, ufPackageUnit);
  4138. AddFlagTextRes(Result, @RsPePkgWeak, UnitFlags, ufWeakUnit);
  4139. AddFlagTextRes(Result, @RsPePkgOrgWeak, UnitFlags, ufOrgWeakUnit);
  4140. AddFlagTextRes(Result, @RsPePkgImplicit, UnitFlags, ufImplicitUnit);
  4141. end;
  4142. //=== { TJclPeBorForm } ======================================================
  4143. constructor TJclPeBorForm.Create(AResItem: TJclPeResourceItem;
  4144. AFormFlags: TFilerFlags; AFormPosition: Integer;
  4145. const AFormClassName, AFormObjectName: string);
  4146. begin
  4147. inherited Create;
  4148. FResItem := AResItem;
  4149. FFormFlags := AFormFlags;
  4150. FFormPosition := AFormPosition;
  4151. FFormClassName := AFormClassName;
  4152. FFormObjectName := AFormObjectName;
  4153. end;
  4154. procedure TJclPeBorForm.ConvertFormToText(const Stream: TStream);
  4155. var
  4156. SourceStream: TJclPeResourceRawStream;
  4157. begin
  4158. SourceStream := TJclPeResourceRawStream.Create(ResItem);
  4159. try
  4160. ObjectBinaryToText(SourceStream, Stream);
  4161. finally
  4162. SourceStream.Free;
  4163. end;
  4164. end;
  4165. procedure TJclPeBorForm.ConvertFormToText(const Strings: TStrings);
  4166. var
  4167. TempStream: TMemoryStream;
  4168. begin
  4169. TempStream := TMemoryStream.Create;
  4170. try
  4171. ConvertFormToText(TempStream);
  4172. TempStream.Seek(0, soFromBeginning);
  4173. Strings.LoadFromStream(TempStream);
  4174. finally
  4175. TempStream.Free;
  4176. end;
  4177. end;
  4178. function TJclPeBorForm.GetDisplayName: string;
  4179. begin
  4180. if FFormObjectName <> '' then
  4181. Result := FFormObjectName + ': '
  4182. else
  4183. Result := '';
  4184. Result := Result + FFormClassName;
  4185. end;
  4186. //=== { TJclPeBorImage } =====================================================
  4187. constructor TJclPeBorImage.Create(ANoExceptions: Boolean);
  4188. begin
  4189. FForms := TObjectList.Create(True);
  4190. FPackageInfoSorted := True;
  4191. inherited Create(ANoExceptions);
  4192. end;
  4193. destructor TJclPeBorImage.Destroy;
  4194. begin
  4195. inherited Destroy;
  4196. FreeAndNil(FForms);
  4197. end;
  4198. procedure TJclPeBorImage.AfterOpen;
  4199. var
  4200. HasDVCLAL, HasPACKAGEINFO, HasPACKAGEOPTIONS: Boolean;
  4201. begin
  4202. inherited AfterOpen;
  4203. if StatusOK then
  4204. with ResourceList do
  4205. begin
  4206. HasDVCLAL := (FindResource(rtRCData, DVclAlResName) <> nil);
  4207. HasPACKAGEINFO := (FindResource(rtRCData, PackageInfoResName) <> nil);
  4208. HasPACKAGEOPTIONS := (FindResource(rtRCData, PackageOptionsResName) <> nil);
  4209. FIsPackage := HasPACKAGEINFO and HasPACKAGEOPTIONS;
  4210. FIsBorlandImage := HasDVCLAL or FIsPackage;
  4211. end;
  4212. end;
  4213. procedure TJclPeBorImage.Clear;
  4214. begin
  4215. FForms.Clear;
  4216. FreeAndNil(FPackageInfo);
  4217. FreeLibHandle;
  4218. inherited Clear;
  4219. FIsBorlandImage := False;
  4220. FIsPackage := False;
  4221. FPackageCompilerVersion := 0;
  4222. end;
  4223. procedure TJclPeBorImage.CreateFormsList;
  4224. var
  4225. ResTypeItem: TJclPeResourceItem;
  4226. I: Integer;
  4227. procedure ProcessListItem(DfmResItem: TJclPeResourceItem);
  4228. const
  4229. FilerSignature: array [1..4] of AnsiChar = string('TPF0');
  4230. var
  4231. SourceStream: TJclPeResourceRawStream;
  4232. Reader: TReader;
  4233. FormFlags: TFilerFlags;
  4234. FormPosition: Integer;
  4235. ClassName, FormName: string;
  4236. begin
  4237. SourceStream := TJclPeResourceRawStream.Create(DfmResItem);
  4238. try
  4239. if (SourceStream.Size > SizeOf(FilerSignature)) and
  4240. (PInteger(SourceStream.Memory)^ = Integer(FilerSignature)) then
  4241. begin
  4242. Reader := TReader.Create(SourceStream, 4096);
  4243. try
  4244. Reader.ReadSignature;
  4245. Reader.ReadPrefix(FormFlags, FormPosition);
  4246. ClassName := Reader.ReadStr;
  4247. FormName := Reader.ReadStr;
  4248. FForms.Add(TJclPeBorForm.Create(DfmResItem, FormFlags, FormPosition,
  4249. ClassName, FormName));
  4250. finally
  4251. Reader.Free;
  4252. end;
  4253. end;
  4254. finally
  4255. SourceStream.Free;
  4256. end;
  4257. end;
  4258. begin
  4259. if StatusOK then
  4260. with ResourceList do
  4261. begin
  4262. ResTypeItem := FindResource(rtRCData, '');
  4263. if ResTypeItem <> nil then
  4264. with ResTypeItem.List do
  4265. for I := 0 to Count - 1 do
  4266. ProcessListItem(Items[I].List[0]);
  4267. end;
  4268. end;
  4269. function TJclPeBorImage.DependedPackages(List: TStrings; FullPathName, Descriptions: Boolean): Boolean;
  4270. var
  4271. ImportList: TStringList;
  4272. I: Integer;
  4273. Name: string;
  4274. begin
  4275. Result := IsBorlandImage;
  4276. if not Result then
  4277. Exit;
  4278. ImportList := InternalImportedLibraries(FileName, True, FullPathName, nil);
  4279. List.BeginUpdate;
  4280. try
  4281. for I := 0 to ImportList.Count - 1 do
  4282. begin
  4283. Name := ImportList[I];
  4284. if StrSame(ExtractFileExt(Name), BinaryExtensionPackage) then
  4285. begin
  4286. if Descriptions then
  4287. List.Add(Name + '=' + GetPackageDescription(PChar(Name)))
  4288. else
  4289. List.Add(Name);
  4290. end;
  4291. end;
  4292. finally
  4293. ImportList.Free;
  4294. List.EndUpdate;
  4295. end;
  4296. end;
  4297. function TJclPeBorImage.FreeLibHandle: Boolean;
  4298. begin
  4299. if FLibHandle <> 0 then
  4300. begin
  4301. Result := FreeLibrary(FLibHandle);
  4302. FLibHandle := 0;
  4303. end
  4304. else
  4305. Result := True;
  4306. end;
  4307. function TJclPeBorImage.GetFormCount: Integer;
  4308. begin
  4309. if FForms.Count = 0 then
  4310. CreateFormsList;
  4311. Result := FForms.Count;
  4312. end;
  4313. function TJclPeBorImage.GetFormFromName(const FormClassName: string): TJclPeBorForm;
  4314. var
  4315. I: Integer;
  4316. begin
  4317. Result := nil;
  4318. for I := 0 to FormCount - 1 do
  4319. if StrSame(FormClassName, Forms[I].FormClassName) then
  4320. begin
  4321. Result := Forms[I];
  4322. Break;
  4323. end;
  4324. end;
  4325. function TJclPeBorImage.GetForms(Index: Integer): TJclPeBorForm;
  4326. begin
  4327. Result := TJclPeBorForm(FForms[Index]);
  4328. end;
  4329. function TJclPeBorImage.GetLibHandle: THandle;
  4330. begin
  4331. if StatusOK and (FLibHandle = 0) then
  4332. begin
  4333. FLibHandle := LoadLibraryEx(PChar(FileName), 0, LOAD_LIBRARY_AS_DATAFILE);
  4334. if FLibHandle = 0 then
  4335. RaiseLastOSError;
  4336. end;
  4337. Result := FLibHandle;
  4338. end;
  4339. function TJclPeBorImage.GetPackageCompilerVersion: Integer;
  4340. var
  4341. I: Integer;
  4342. ImportName: string;
  4343. function CheckName: Boolean;
  4344. begin
  4345. Result := False;
  4346. ImportName := AnsiUpperCase(ImportName);
  4347. if StrSame(ExtractFileExt(ImportName), BinaryExtensionPackage) then
  4348. begin
  4349. ImportName := PathExtractFileNameNoExt(ImportName);
  4350. if (Length(ImportName) = 5) and
  4351. CharIsDigit(ImportName[4]) and CharIsDigit(ImportName[5]) and
  4352. ((Pos('RTL', ImportName) = 1) or (Pos('VCL', ImportName) = 1)) then
  4353. begin
  4354. FPackageCompilerVersion := StrToIntDef(Copy(ImportName, 4, 2), 0);
  4355. Result := True;
  4356. end;
  4357. end;
  4358. end;
  4359. begin
  4360. if (FPackageCompilerVersion = 0) and IsPackage then
  4361. begin
  4362. with ImportList do
  4363. for I := 0 to UniqueLibItemCount - 1 do
  4364. begin
  4365. ImportName := UniqueLibNames[I];
  4366. if CheckName then
  4367. Break;
  4368. end;
  4369. if FPackageCompilerVersion = 0 then
  4370. begin
  4371. ImportName := ExtractFileName(FileName);
  4372. CheckName;
  4373. end;
  4374. end;
  4375. Result := FPackageCompilerVersion;
  4376. end;
  4377. function TJclPeBorImage.GetPackageInfo: TJclPePackageInfo;
  4378. begin
  4379. if StatusOK and (FPackageInfo = nil) then
  4380. begin
  4381. GetLibHandle;
  4382. FPackageInfo := TJclPePackageInfo.Create(FLibHandle);
  4383. FPackageInfo.Sorted := FPackageInfoSorted;
  4384. FreeLibHandle;
  4385. end;
  4386. Result := FPackageInfo;
  4387. end;
  4388. {$ENDIF BORLAND}
  4389. //=== { TJclPeNameSearch } ===================================================
  4390. constructor TJclPeNameSearch.Create(const FunctionName, Path: string; Options: TJclPeNameSearchOptions);
  4391. begin
  4392. inherited Create(True);
  4393. FFunctionName := FunctionName;
  4394. FOptions := Options;
  4395. FPath := Path;
  4396. FreeOnTerminate := True;
  4397. end;
  4398. function TJclPeNameSearch.CompareName(const FunctionName, ComparedName: string): Boolean;
  4399. begin
  4400. Result := PeSmartFunctionNameSame(ComparedName, FunctionName, [scIgnoreCase]);
  4401. end;
  4402. procedure TJclPeNameSearch.DoFound;
  4403. begin
  4404. if Assigned(FOnFound) then
  4405. FOnFound(Self, F_FileName, F_FunctionName, F_Option);
  4406. end;
  4407. procedure TJclPeNameSearch.DoProcessFile;
  4408. begin
  4409. if Assigned(FOnProcessFile) then
  4410. FOnProcessFile(Self, FPeImage, F_Process);
  4411. end;
  4412. procedure TJclPeNameSearch.Execute;
  4413. var
  4414. PathList: TStringList;
  4415. I: Integer;
  4416. function CompareNameAndNotify(const S: string): Boolean;
  4417. begin
  4418. Result := CompareName(S, FFunctionName);
  4419. if Result and not Terminated then
  4420. begin
  4421. F_FunctionName := S;
  4422. Synchronize(DoFound);
  4423. end;
  4424. end;
  4425. procedure ProcessDirectorySearch(const DirName: string);
  4426. var
  4427. Se: TSearchRec;
  4428. SearchResult: Integer;
  4429. ImportList: TJclPeImportList;
  4430. ExportList: TJclPeExportFuncList;
  4431. I: Integer;
  4432. begin
  4433. SearchResult := FindFirst(DirName, faArchive + faReadOnly, Se);
  4434. try
  4435. while not Terminated and (SearchResult = 0) do
  4436. begin
  4437. F_FileName := PathAddSeparator(ExtractFilePath(DirName)) + Se.Name;
  4438. F_Process := True;
  4439. FPeImage.FileName := F_FileName;
  4440. if Assigned(FOnProcessFile) then
  4441. Synchronize(DoProcessFile);
  4442. if F_Process and FPeImage.StatusOK then
  4443. begin
  4444. if seExports in FOptions then
  4445. begin
  4446. ExportList := FPeImage.ExportList;
  4447. F_Option := seExports;
  4448. for I := 0 to ExportList.Count - 1 do
  4449. begin
  4450. if Terminated then
  4451. Break;
  4452. CompareNameAndNotify(ExportList[I].Name);
  4453. end;
  4454. end;
  4455. if FOptions * [seImports, seDelayImports, seBoundImports] <> [] then
  4456. begin
  4457. ImportList := FPeImage.ImportList;
  4458. FPeImage.TryGetNamesForOrdinalImports;
  4459. for I := 0 to ImportList.AllItemCount - 1 do
  4460. with ImportList.AllItems[I] do
  4461. begin
  4462. if Terminated then
  4463. Break;
  4464. case ImportLib.ImportKind of
  4465. ikImport:
  4466. if seImports in FOptions then
  4467. begin
  4468. F_Option := seImports;
  4469. CompareNameAndNotify(Name);
  4470. end;
  4471. ikDelayImport:
  4472. if seDelayImports in FOptions then
  4473. begin
  4474. F_Option := seDelayImports;
  4475. CompareNameAndNotify(Name);
  4476. end;
  4477. ikBoundImport:
  4478. if seDelayImports in FOptions then
  4479. begin
  4480. F_Option := seBoundImports;
  4481. CompareNameAndNotify(Name);
  4482. end;
  4483. end;
  4484. end;
  4485. end;
  4486. end;
  4487. SearchResult := FindNext(Se);
  4488. end;
  4489. finally
  4490. FindClose(Se);
  4491. end;
  4492. end;
  4493. begin
  4494. FPeImage := TJclPeImage.Create(True);
  4495. PathList := TStringList.Create;
  4496. try
  4497. PathList.Sorted := True;
  4498. PathList.Duplicates := dupIgnore;
  4499. StrToStrings(FPath, ';', PathList);
  4500. for I := 0 to PathList.Count - 1 do
  4501. ProcessDirectorySearch(PathAddSeparator(Trim(PathList[I])) + '*.*');
  4502. finally
  4503. PathList.Free;
  4504. FPeImage.Free;
  4505. end;
  4506. end;
  4507. procedure TJclPeNameSearch.Start;
  4508. begin
  4509. {$IFDEF RTL210_UP}
  4510. Suspended := False;
  4511. {$ELSE ~RTL210_UP}
  4512. Resume;
  4513. {$ENDIF ~RTL210_UP}
  4514. end;
  4515. //=== PE Image miscellaneous functions =======================================
  4516. function IsValidPeFile(const FileName: TFileName): Boolean;
  4517. var
  4518. NtHeaders: TImageNtHeaders32;
  4519. begin
  4520. Result := PeGetNtHeaders32(FileName, NtHeaders);
  4521. end;
  4522. function InternalGetNtHeaders32(const FileName: TFileName; out NtHeaders): Boolean;
  4523. var
  4524. FileHandle: THandle;
  4525. Mapping: TJclFileMapping;
  4526. View: TJclFileMappingView;
  4527. HeadersPtr: PImageNtHeaders32;
  4528. begin
  4529. Result := False;
  4530. ResetMemory(NtHeaders, SizeOf(TImageNtHeaders32));
  4531. FileHandle := FileOpen(FileName, fmOpenRead or fmShareDenyWrite);
  4532. if FileHandle = INVALID_HANDLE_VALUE then
  4533. Exit;
  4534. try
  4535. if GetSizeOfFile(FileHandle) >= SizeOf(TImageDosHeader) then
  4536. begin
  4537. Mapping := TJclFileMapping.Create(FileHandle, '', PAGE_READONLY, 0, nil);
  4538. try
  4539. View := TJclFileMappingView.Create(Mapping, FILE_MAP_READ, 0, 0);
  4540. HeadersPtr := PeMapImgNtHeaders32(View.Memory);
  4541. if HeadersPtr <> nil then
  4542. begin
  4543. Result := True;
  4544. TImageNtHeaders32(NtHeaders) := HeadersPtr^;
  4545. end;
  4546. finally
  4547. Mapping.Free;
  4548. end;
  4549. end;
  4550. finally
  4551. FileClose(FileHandle);
  4552. end;
  4553. end;
  4554. function PeGetNtHeaders32(const FileName: TFileName; out NtHeaders: TImageNtHeaders32): Boolean;
  4555. begin
  4556. Result := InternalGetNtHeaders32(FileName, NtHeaders);
  4557. end;
  4558. function PeGetNtHeaders64(const FileName: TFileName; out NtHeaders: TImageNtHeaders64): Boolean;
  4559. var
  4560. FileHandle: THandle;
  4561. Mapping: TJclFileMapping;
  4562. View: TJclFileMappingView;
  4563. HeadersPtr: PImageNtHeaders64;
  4564. begin
  4565. Result := False;
  4566. ResetMemory(NtHeaders, SizeOf(NtHeaders));
  4567. FileHandle := FileOpen(FileName, fmOpenRead or fmShareDenyWrite);
  4568. if FileHandle = INVALID_HANDLE_VALUE then
  4569. Exit;
  4570. try
  4571. if GetSizeOfFile(FileHandle) >= SizeOf(TImageDosHeader) then
  4572. begin
  4573. Mapping := TJclFileMapping.Create(FileHandle, '', PAGE_READONLY, 0, nil);
  4574. try
  4575. View := TJclFileMappingView.Create(Mapping, FILE_MAP_READ, 0, 0);
  4576. HeadersPtr := PeMapImgNtHeaders64(View.Memory);
  4577. if HeadersPtr <> nil then
  4578. begin
  4579. Result := True;
  4580. NtHeaders := HeadersPtr^;
  4581. end;
  4582. finally
  4583. Mapping.Free;
  4584. end;
  4585. end;
  4586. finally
  4587. FileClose(FileHandle);
  4588. end;
  4589. end;
  4590. function PeCreateNameHintTable(const FileName: TFileName): Boolean;
  4591. var
  4592. PeImage, ExportsImage: TJclPeImage;
  4593. I: Integer;
  4594. ImportItem: TJclPeImportLibItem;
  4595. Thunk32: PImageThunkData32;
  4596. Thunk64: PImageThunkData64;
  4597. OrdinalName: PImageImportByName;
  4598. ExportItem: TJclPeExportFuncItem;
  4599. Cache: TJclPeImagesCache;
  4600. ImageBase32: TJclAddr32;
  4601. ImageBase64: TJclAddr64;
  4602. UTF8Name: TUTF8String;
  4603. ExportName: string;
  4604. begin
  4605. Cache := TJclPeImagesCache.Create;
  4606. try
  4607. PeImage := TJclPeImage.Create(False);
  4608. try
  4609. PeImage.ReadOnlyAccess := False;
  4610. PeImage.FileName := FileName;
  4611. Result := PeImage.ImportList.Count > 0;
  4612. for I := 0 to PeImage.ImportList.Count - 1 do
  4613. begin
  4614. ImportItem := PeImage.ImportList[I];
  4615. if ImportItem.ImportKind = ikBoundImport then
  4616. Continue;
  4617. ExportsImage := Cache[ImportItem.FileName];
  4618. ExportsImage.ExportList.PrepareForFastNameSearch;
  4619. case PEImage.Target of
  4620. taWin32:
  4621. begin
  4622. Thunk32 := ImportItem.ThunkData32;
  4623. ImageBase32 := PeImage.OptionalHeader32.ImageBase;
  4624. while Thunk32^.Function_ <> 0 do
  4625. begin
  4626. if Thunk32^.Ordinal and IMAGE_ORDINAL_FLAG32 = 0 then
  4627. begin
  4628. case ImportItem.ImportKind of
  4629. ikImport:
  4630. OrdinalName := PImageImportByName(PeImage.RvaToVa(Thunk32^.AddressOfData));
  4631. ikDelayImport:
  4632. OrdinalName := PImageImportByName(PeImage.RvaToVa(Thunk32^.AddressOfData - ImageBase32));
  4633. else
  4634. OrdinalName := nil;
  4635. end;
  4636. UTF8Name := PAnsiChar(@OrdinalName.Name);
  4637. if not TryUTF8ToString(UTF8Name, ExportName) then
  4638. ExportName := string(UTF8Name);
  4639. ExportItem := ExportsImage.ExportList.ItemFromName[ExportName];
  4640. if ExportItem <> nil then
  4641. OrdinalName.Hint := ExportItem.Hint
  4642. else
  4643. OrdinalName.Hint := 0;
  4644. end;
  4645. Inc(Thunk32);
  4646. end;
  4647. end;
  4648. taWin64:
  4649. begin
  4650. Thunk64 := ImportItem.ThunkData64;
  4651. ImageBase64 := PeImage.OptionalHeader64.ImageBase;
  4652. while Thunk64^.Function_ <> 0 do
  4653. begin
  4654. if Thunk64^.Ordinal and IMAGE_ORDINAL_FLAG64 = 0 then
  4655. begin
  4656. case ImportItem.ImportKind of
  4657. ikImport:
  4658. OrdinalName := PImageImportByName(PeImage.RvaToVa(Thunk64^.AddressOfData));
  4659. ikDelayImport:
  4660. OrdinalName := PImageImportByName(PeImage.RvaToVa(Thunk64^.AddressOfData - ImageBase64));
  4661. else
  4662. OrdinalName := nil;
  4663. end;
  4664. UTF8Name := PAnsiChar(@OrdinalName.Name);
  4665. if not TryUTF8ToString(UTF8Name, ExportName) then
  4666. ExportName := string(UTF8Name);
  4667. ExportItem := ExportsImage.ExportList.ItemFromName[ExportName];
  4668. if ExportItem <> nil then
  4669. OrdinalName.Hint := ExportItem.Hint
  4670. else
  4671. OrdinalName.Hint := 0;
  4672. end;
  4673. Inc(Thunk64);
  4674. end;
  4675. end;
  4676. end;
  4677. end;
  4678. finally
  4679. PeImage.Free;
  4680. end;
  4681. finally
  4682. Cache.Free;
  4683. end;
  4684. end;
  4685. function PeRebaseImage32(const ImageName: TFileName; NewBase: TJclAddr32;
  4686. TimeStamp, MaxNewSize: DWORD): TJclRebaseImageInfo32;
  4687. function CalculateBaseAddress: TJclAddr32;
  4688. var
  4689. FirstChar: Char;
  4690. ModuleName: string;
  4691. begin
  4692. ModuleName := ExtractFileName(ImageName);
  4693. if Length(ModuleName) > 0 then
  4694. FirstChar := UpCase(ModuleName[1])
  4695. else
  4696. FirstChar := NativeNull;
  4697. if not CharIsUpper(FirstChar) then
  4698. FirstChar := 'A';
  4699. Result := $60000000 + (((Ord(FirstChar) - Ord('A')) div 3) * $1000000);
  4700. end;
  4701. {$IFDEF CPU64}
  4702. {$IFNDEF DELPHI64_TEMPORARY}
  4703. var
  4704. NewIB, OldIB: QWord;
  4705. {$ENDIF CPU64}
  4706. {$ENDIF ~DELPHI64_TEMPORARY}
  4707. begin
  4708. if NewBase = 0 then
  4709. NewBase := CalculateBaseAddress;
  4710. with Result do
  4711. begin
  4712. NewImageBase := NewBase;
  4713. // OF: possible loss of data
  4714. {$IFDEF CPU32}
  4715. Win32Check(ReBaseImage(PAnsiChar(AnsiString(ImageName)), nil, True, False, False, MaxNewSize,
  4716. OldImageSize, OldImageBase, NewImageSize, NewImageBase, TimeStamp));
  4717. {$ENDIF CPU32}
  4718. {$IFDEF CPU64}
  4719. {$IFDEF DELPHI64_TEMPORARY}
  4720. System.Error(rePlatformNotImplemented);
  4721. {$ELSE ~DELPHI64_TEMPORARY}
  4722. NewIB := NewImageBase;
  4723. OldIB := OldImageBase;
  4724. Win32Check(ReBaseImage(PAnsiChar(AnsiString(ImageName)), nil, True, False, False, MaxNewSize,
  4725. OldImageSize, OldIB, NewImageSize, NewIB, TimeStamp));
  4726. NewImageBase := NewIB;
  4727. OldImageBase := OldIB;
  4728. {$ENDIF ~DELPHI64_TEMPORARY}
  4729. {$ENDIF CPU64}
  4730. end;
  4731. end;
  4732. function PeRebaseImage64(const ImageName: TFileName; NewBase: TJclAddr64;
  4733. TimeStamp, MaxNewSize: DWORD): TJclRebaseImageInfo64;
  4734. function CalculateBaseAddress: TJclAddr64;
  4735. var
  4736. FirstChar: Char;
  4737. ModuleName: string;
  4738. begin
  4739. ModuleName := ExtractFileName(ImageName);
  4740. if Length(ModuleName) > 0 then
  4741. FirstChar := UpCase(ModuleName[1])
  4742. else
  4743. FirstChar := NativeNull;
  4744. if not CharIsUpper(FirstChar) then
  4745. FirstChar := 'A';
  4746. Result := $60000000 + (((Ord(FirstChar) - Ord('A')) div 3) * $1000000);
  4747. Result := Result shl 32;
  4748. end;
  4749. begin
  4750. if NewBase = 0 then
  4751. NewBase := CalculateBaseAddress;
  4752. with Result do
  4753. begin
  4754. NewImageBase := NewBase;
  4755. // OF: possible loss of data
  4756. Win32Check(ReBaseImage64(PAnsiChar(AnsiString(ImageName)), nil, True, False, False, MaxNewSize,
  4757. OldImageSize, OldImageBase, NewImageSize, NewImageBase, TimeStamp));
  4758. end;
  4759. end;
  4760. function PeUpdateLinkerTimeStamp(const FileName: TFileName; const Time: TDateTime): Boolean;
  4761. var
  4762. Mapping: TJclFileMapping;
  4763. View: TJclFileMappingView;
  4764. Headers: PImageNtHeaders32; // works with 64-bit binaries too
  4765. // only the optional field differs
  4766. begin
  4767. Mapping := TJclFileMapping.Create(FileName, fmOpenReadWrite, '', PAGE_READWRITE, 0, nil);
  4768. try
  4769. View := TJclFileMappingView.Create(Mapping, FILE_MAP_WRITE, 0, 0);
  4770. Headers := PeMapImgNtHeaders32(View.Memory);
  4771. Result := (Headers <> nil);
  4772. if Result then
  4773. Headers^.FileHeader.TimeDateStamp := TJclPeImage.DateTimeToStamp(Time);
  4774. finally
  4775. Mapping.Free;
  4776. end;
  4777. end;
  4778. function PeReadLinkerTimeStamp(const FileName: TFileName): TDateTime;
  4779. var
  4780. Mapping: TJclFileMappingStream;
  4781. Headers: PImageNtHeaders32; // works with 64-bit binaries too
  4782. // only the optional field differs
  4783. begin
  4784. Mapping := TJclFileMappingStream.Create(FileName, fmOpenRead or fmShareDenyWrite);
  4785. try
  4786. Headers := PeMapImgNtHeaders32(Mapping.Memory);
  4787. if Headers <> nil then
  4788. Result := TJclPeImage.StampToDateTime(Headers^.FileHeader.TimeDateStamp)
  4789. else
  4790. Result := -1;
  4791. finally
  4792. Mapping.Free;
  4793. end;
  4794. end;
  4795. { TODO -cHelp : Author: Uwe Schuster(just a generic version of JclDebug.InsertDebugDataIntoExecutableFile) }
  4796. function PeInsertSection(const FileName: TFileName; SectionStream: TStream; SectionName: string): Boolean;
  4797. procedure RoundUpToAlignment(var Value: DWORD; Alignment: DWORD);
  4798. begin
  4799. if (Value mod Alignment) <> 0 then
  4800. Value := ((Value div Alignment) + 1) * Alignment;
  4801. end;
  4802. function PeInsertSection32(ImageStream: TMemoryStream): Boolean;
  4803. var
  4804. NtHeaders: PImageNtHeaders32;
  4805. Sections, LastSection, NewSection: PImageSectionHeader;
  4806. VirtualAlignedSize: DWORD;
  4807. I, X, NeedFill: Integer;
  4808. SectionDataSize: Integer;
  4809. UTF8Name: TUTF8String;
  4810. begin
  4811. Result := True;
  4812. try
  4813. SectionDataSize := SectionStream.Size;
  4814. NtHeaders := PeMapImgNtHeaders32(ImageStream.Memory);
  4815. Assert(NtHeaders <> nil);
  4816. Sections := PeMapImgSections32(NtHeaders);
  4817. Assert(Sections <> nil);
  4818. // Check whether there is not a section with the name already. If so, return True (#0000069)
  4819. if PeMapImgFindSection32(NtHeaders, SectionName) <> nil then
  4820. begin
  4821. Result := True;
  4822. Exit;
  4823. end;
  4824. LastSection := Sections;
  4825. Inc(LastSection, NtHeaders^.FileHeader.NumberOfSections - 1);
  4826. NewSection := LastSection;
  4827. Inc(NewSection);
  4828. // Increase the number of sections
  4829. Inc(NtHeaders^.FileHeader.NumberOfSections);
  4830. ResetMemory(NewSection^, SizeOf(TImageSectionHeader));
  4831. // JCLDEBUG Virtual Address
  4832. NewSection^.VirtualAddress := LastSection^.VirtualAddress + LastSection^.Misc.VirtualSize;
  4833. RoundUpToAlignment(NewSection^.VirtualAddress, NtHeaders^.OptionalHeader.SectionAlignment);
  4834. // JCLDEBUG Physical Offset
  4835. NewSection^.PointerToRawData := LastSection^.PointerToRawData + LastSection^.SizeOfRawData;
  4836. RoundUpToAlignment(NewSection^.PointerToRawData, NtHeaders^.OptionalHeader.FileAlignment);
  4837. // JCLDEBUG Section name
  4838. if not TryStringToUTF8(SectionName, UTF8Name) then
  4839. UTF8Name := TUTF8String(SectionName);
  4840. StrPLCopyA(PAnsiChar(@NewSection^.Name), UTF8Name, IMAGE_SIZEOF_SHORT_NAME);
  4841. // JCLDEBUG Characteristics flags
  4842. NewSection^.Characteristics := IMAGE_SCN_MEM_READ or IMAGE_SCN_CNT_INITIALIZED_DATA;
  4843. // Size of virtual data area
  4844. NewSection^.Misc.VirtualSize := SectionDataSize;
  4845. VirtualAlignedSize := SectionDataSize;
  4846. RoundUpToAlignment(VirtualAlignedSize, NtHeaders^.OptionalHeader.SectionAlignment);
  4847. // Update Size of Image
  4848. Inc(NtHeaders^.OptionalHeader.SizeOfImage, VirtualAlignedSize);
  4849. // Raw data size
  4850. NewSection^.SizeOfRawData := SectionDataSize;
  4851. RoundUpToAlignment(NewSection^.SizeOfRawData, NtHeaders^.OptionalHeader.FileAlignment);
  4852. // Update Initialized data size
  4853. Inc(NtHeaders^.OptionalHeader.SizeOfInitializedData, NewSection^.SizeOfRawData);
  4854. // Fill data to alignment
  4855. NeedFill := INT_PTR(NewSection^.SizeOfRawData) - SectionDataSize;
  4856. // Note: Delphi linker seems to generate incorrect (unaligned) size of
  4857. // the executable when adding TD32 debug data so the position could be
  4858. // behind the size of the file then.
  4859. ImageStream.Seek(NewSection^.PointerToRawData, soBeginning);
  4860. ImageStream.CopyFrom(SectionStream, 0);
  4861. X := 0;
  4862. for I := 1 to NeedFill do
  4863. ImageStream.WriteBuffer(X, 1);
  4864. except
  4865. Result := False;
  4866. end;
  4867. end;
  4868. function PeInsertSection64(ImageStream: TMemoryStream): Boolean;
  4869. var
  4870. NtHeaders: PImageNtHeaders64;
  4871. Sections, LastSection, NewSection: PImageSectionHeader;
  4872. VirtualAlignedSize: DWORD;
  4873. I, X, NeedFill: Integer;
  4874. SectionDataSize: Integer;
  4875. UTF8Name: TUTF8String;
  4876. begin
  4877. Result := True;
  4878. try
  4879. SectionDataSize := SectionStream.Size;
  4880. NtHeaders := PeMapImgNtHeaders64(ImageStream.Memory);
  4881. Assert(NtHeaders <> nil);
  4882. Sections := PeMapImgSections64(NtHeaders);
  4883. Assert(Sections <> nil);
  4884. // Check whether there is not a section with the name already. If so, return True (#0000069)
  4885. if PeMapImgFindSection64(NtHeaders, SectionName) <> nil then
  4886. begin
  4887. Result := True;
  4888. Exit;
  4889. end;
  4890. LastSection := Sections;
  4891. Inc(LastSection, NtHeaders^.FileHeader.NumberOfSections - 1);
  4892. NewSection := LastSection;
  4893. Inc(NewSection);
  4894. // Increase the number of sections
  4895. Inc(NtHeaders^.FileHeader.NumberOfSections);
  4896. ResetMemory(NewSection^, SizeOf(TImageSectionHeader));
  4897. // JCLDEBUG Virtual Address
  4898. NewSection^.VirtualAddress := LastSection^.VirtualAddress + LastSection^.Misc.VirtualSize;
  4899. RoundUpToAlignment(NewSection^.VirtualAddress, NtHeaders^.OptionalHeader.SectionAlignment);
  4900. // JCLDEBUG Physical Offset
  4901. NewSection^.PointerToRawData := LastSection^.PointerToRawData + LastSection^.SizeOfRawData;
  4902. RoundUpToAlignment(NewSection^.PointerToRawData, NtHeaders^.OptionalHeader.FileAlignment);
  4903. // JCLDEBUG Section name
  4904. if not TryStringToUTF8(SectionName, UTF8Name) then
  4905. UTF8Name := TUTF8String(SectionName);
  4906. StrPLCopyA(PAnsiChar(@NewSection^.Name), UTF8Name, IMAGE_SIZEOF_SHORT_NAME);
  4907. // JCLDEBUG Characteristics flags
  4908. NewSection^.Characteristics := IMAGE_SCN_MEM_READ or IMAGE_SCN_CNT_INITIALIZED_DATA;
  4909. // Size of virtual data area
  4910. NewSection^.Misc.VirtualSize := SectionDataSize;
  4911. VirtualAlignedSize := SectionDataSize;
  4912. RoundUpToAlignment(VirtualAlignedSize, NtHeaders^.OptionalHeader.SectionAlignment);
  4913. // Update Size of Image
  4914. Inc(NtHeaders^.OptionalHeader.SizeOfImage, VirtualAlignedSize);
  4915. // Raw data size
  4916. NewSection^.SizeOfRawData := SectionDataSize;
  4917. RoundUpToAlignment(NewSection^.SizeOfRawData, NtHeaders^.OptionalHeader.FileAlignment);
  4918. // Update Initialized data size
  4919. Inc(NtHeaders^.OptionalHeader.SizeOfInitializedData, NewSection^.SizeOfRawData);
  4920. // Fill data to alignment
  4921. NeedFill := INT_PTR(NewSection^.SizeOfRawData) - SectionDataSize;
  4922. // Note: Delphi linker seems to generate incorrect (unaligned) size of
  4923. // the executable when adding TD32 debug data so the position could be
  4924. // behind the size of the file then.
  4925. ImageStream.Seek(NewSection^.PointerToRawData, soBeginning);
  4926. ImageStream.CopyFrom(SectionStream, 0);
  4927. X := 0;
  4928. for I := 1 to NeedFill do
  4929. ImageStream.WriteBuffer(X, 1);
  4930. except
  4931. Result := False;
  4932. end;
  4933. end;
  4934. var
  4935. ImageStream: TMemoryStream;
  4936. begin
  4937. Result := Assigned(SectionStream) and (SectionName <> '');
  4938. if not Result then
  4939. Exit;
  4940. ImageStream := TMemoryStream.Create;
  4941. try
  4942. ImageStream.LoadFromFile(FileName);
  4943. case PeMapImgTarget(ImageStream.Memory) of
  4944. taWin32:
  4945. Result := PeInsertSection32(ImageStream);
  4946. taWin64:
  4947. Result := PeInsertSection64(ImageStream);
  4948. //taUnknown:
  4949. else
  4950. Result := False;
  4951. end;
  4952. if Result then
  4953. ImageStream.SaveToFile(FileName);
  4954. finally
  4955. ImageStream.Free;
  4956. end;
  4957. end;
  4958. function PeVerifyCheckSum(const FileName: TFileName): Boolean;
  4959. begin
  4960. with CreatePeImage(FileName) do
  4961. try
  4962. Result := VerifyCheckSum;
  4963. finally
  4964. Free;
  4965. end;
  4966. end;
  4967. function PeClearCheckSum(const FileName: TFileName): Boolean;
  4968. function PeClearCheckSum32(ModuleAddress: Pointer): Boolean;
  4969. var
  4970. Headers: PImageNtHeaders32;
  4971. begin
  4972. Headers := PeMapImgNtHeaders32(ModuleAddress);
  4973. Result := (Headers <> nil);
  4974. if Result then
  4975. Headers^.OptionalHeader.CheckSum := 0;
  4976. end;
  4977. function PeClearCheckSum64(ModuleAddress: Pointer): Boolean;
  4978. var
  4979. Headers: PImageNtHeaders64;
  4980. begin
  4981. Headers := PeMapImgNtHeaders64(ModuleAddress);
  4982. Result := (Headers <> nil);
  4983. if Result then
  4984. Headers^.OptionalHeader.CheckSum := 0;
  4985. end;
  4986. var
  4987. Mapping: TJclFileMapping;
  4988. View: TJclFileMappingView;
  4989. begin
  4990. Mapping := TJclFileMapping.Create(FileName, fmOpenReadWrite, '', PAGE_READWRITE, 0, nil);
  4991. try
  4992. View := TJclFileMappingView.Create(Mapping, FILE_MAP_WRITE, 0, 0);
  4993. case PeMapImgTarget(View.Memory) of
  4994. taWin32:
  4995. Result := PeClearCheckSum32(View.Memory);
  4996. taWin64:
  4997. Result := PeClearCheckSum64(View.Memory);
  4998. //taUnknown:
  4999. else
  5000. Result := False;
  5001. end;
  5002. finally
  5003. Mapping.Free;
  5004. end;
  5005. end;
  5006. function PeUpdateCheckSum(const FileName: TFileName): Boolean;
  5007. var
  5008. LI: TLoadedImage;
  5009. begin
  5010. LI.ModuleName := nil;
  5011. // OF: possible loss of data
  5012. Result := MapAndLoad(PAnsiChar(AnsiString(FileName)), nil, LI, True, False);
  5013. if Result then
  5014. Result := UnMapAndLoad(LI);
  5015. end;
  5016. // Various simple PE Image searching and listing routines
  5017. function PeDoesExportFunction(const FileName: TFileName; const FunctionName: string;
  5018. Options: TJclSmartCompOptions): Boolean;
  5019. begin
  5020. with CreatePeImage(FileName) do
  5021. try
  5022. Result := StatusOK and Assigned(ExportList.SmartFindName(FunctionName, Options));
  5023. finally
  5024. Free;
  5025. end;
  5026. end;
  5027. function PeIsExportFunctionForwardedEx(const FileName: TFileName; const FunctionName: string;
  5028. out ForwardedName: string; Options: TJclSmartCompOptions): Boolean;
  5029. var
  5030. ExportItem: TJclPeExportFuncItem;
  5031. begin
  5032. with CreatePeImage(FileName) do
  5033. try
  5034. Result := StatusOK;
  5035. if Result then
  5036. begin
  5037. ExportItem := ExportList.SmartFindName(FunctionName, Options);
  5038. if ExportItem <> nil then
  5039. begin
  5040. Result := ExportItem.IsForwarded;
  5041. ForwardedName := ExportItem.ForwardedName;
  5042. end
  5043. else
  5044. begin
  5045. Result := False;
  5046. ForwardedName := '';
  5047. end;
  5048. end;
  5049. finally
  5050. Free;
  5051. end;
  5052. end;
  5053. function PeIsExportFunctionForwarded(const FileName: TFileName; const FunctionName: string;
  5054. Options: TJclSmartCompOptions): Boolean;
  5055. var
  5056. Dummy: string;
  5057. begin
  5058. Result := PeIsExportFunctionForwardedEx(FileName, FunctionName, Dummy, Options);
  5059. end;
  5060. function PeDoesImportFunction(const FileName: TFileName; const FunctionName: string;
  5061. const LibraryName: string; Options: TJclSmartCompOptions): Boolean;
  5062. begin
  5063. with CreatePeImage(FileName) do
  5064. try
  5065. Result := StatusOK;
  5066. if Result then
  5067. with ImportList do
  5068. begin
  5069. TryGetNamesForOrdinalImports;
  5070. Result := SmartFindName(FunctionName, LibraryName, Options) <> nil;
  5071. end;
  5072. finally
  5073. Free;
  5074. end;
  5075. end;
  5076. function PeDoesImportLibrary(const FileName: TFileName; const LibraryName: string;
  5077. Recursive: Boolean): Boolean;
  5078. var
  5079. SL: TStringList;
  5080. begin
  5081. with CreatePeImage(FileName) do
  5082. try
  5083. Result := StatusOK;
  5084. if Result then
  5085. begin
  5086. SL := InternalImportedLibraries(FileName, Recursive, False, nil);
  5087. try
  5088. Result := SL.IndexOf(LibraryName) > -1;
  5089. finally
  5090. SL.Free;
  5091. end;
  5092. end;
  5093. finally
  5094. Free;
  5095. end;
  5096. end;
  5097. function PeImportedLibraries(const FileName: TFileName; const LibrariesList: TStrings;
  5098. Recursive, FullPathName: Boolean): Boolean;
  5099. var
  5100. SL: TStringList;
  5101. begin
  5102. with CreatePeImage(FileName) do
  5103. try
  5104. Result := StatusOK;
  5105. if Result then
  5106. begin
  5107. SL := InternalImportedLibraries(FileName, Recursive, FullPathName, nil);
  5108. try
  5109. LibrariesList.Assign(SL);
  5110. finally
  5111. SL.Free;
  5112. end;
  5113. end;
  5114. finally
  5115. Free;
  5116. end;
  5117. end;
  5118. function PeImportedFunctions(const FileName: TFileName; const FunctionsList: TStrings;
  5119. const LibraryName: string; IncludeLibNames: Boolean): Boolean;
  5120. var
  5121. I: Integer;
  5122. begin
  5123. with CreatePeImage(FileName) do
  5124. try
  5125. Result := StatusOK;
  5126. if Result then
  5127. with ImportList do
  5128. begin
  5129. TryGetNamesForOrdinalImports;
  5130. FunctionsList.BeginUpdate;
  5131. try
  5132. for I := 0 to AllItemCount - 1 do
  5133. with AllItems[I] do
  5134. if ((Length(LibraryName) = 0) or StrSame(ImportLib.Name, LibraryName)) and
  5135. (Name <> '') then
  5136. begin
  5137. if IncludeLibNames then
  5138. FunctionsList.Add(ImportLib.Name + '=' + Name)
  5139. else
  5140. FunctionsList.Add(Name);
  5141. end;
  5142. finally
  5143. FunctionsList.EndUpdate;
  5144. end;
  5145. end;
  5146. finally
  5147. Free;
  5148. end;
  5149. end;
  5150. function PeExportedFunctions(const FileName: TFileName; const FunctionsList: TStrings): Boolean;
  5151. var
  5152. I: Integer;
  5153. begin
  5154. with CreatePeImage(FileName) do
  5155. try
  5156. Result := StatusOK;
  5157. if Result then
  5158. begin
  5159. FunctionsList.BeginUpdate;
  5160. try
  5161. with ExportList do
  5162. for I := 0 to Count - 1 do
  5163. with Items[I] do
  5164. if not IsExportedVariable then
  5165. FunctionsList.Add(Name);
  5166. finally
  5167. FunctionsList.EndUpdate;
  5168. end;
  5169. end;
  5170. finally
  5171. Free;
  5172. end;
  5173. end;
  5174. function PeExportedNames(const FileName: TFileName; const FunctionsList: TStrings): Boolean;
  5175. var
  5176. I: Integer;
  5177. begin
  5178. with CreatePeImage(FileName) do
  5179. try
  5180. Result := StatusOK;
  5181. if Result then
  5182. begin
  5183. FunctionsList.BeginUpdate;
  5184. try
  5185. with ExportList do
  5186. for I := 0 to Count - 1 do
  5187. FunctionsList.Add(Items[I].Name);
  5188. finally
  5189. FunctionsList.EndUpdate;
  5190. end;
  5191. end;
  5192. finally
  5193. Free;
  5194. end;
  5195. end;
  5196. function PeExportedVariables(const FileName: TFileName; const FunctionsList: TStrings): Boolean;
  5197. var
  5198. I: Integer;
  5199. begin
  5200. with CreatePeImage(FileName) do
  5201. try
  5202. Result := StatusOK;
  5203. if Result then
  5204. begin
  5205. FunctionsList.BeginUpdate;
  5206. try
  5207. with ExportList do
  5208. for I := 0 to Count - 1 do
  5209. with Items[I] do
  5210. if IsExportedVariable then
  5211. FunctionsList.AddObject(Name, Pointer(Address));
  5212. finally
  5213. FunctionsList.EndUpdate;
  5214. end;
  5215. end;
  5216. finally
  5217. Free;
  5218. end;
  5219. end;
  5220. function PeResourceKindNames(const FileName: TFileName; ResourceType: TJclPeResourceKind;
  5221. const NamesList: TStrings): Boolean;
  5222. begin
  5223. with CreatePeImage(FileName) do
  5224. try
  5225. Result := StatusOK and ResourceList.ListResourceNames(ResourceType, NamesList);
  5226. finally
  5227. Free;
  5228. end;
  5229. end;
  5230. {$IFDEF BORLAND}
  5231. function PeBorFormNames(const FileName: TFileName; const NamesList: TStrings): Boolean;
  5232. var
  5233. I: Integer;
  5234. BorImage: TJclPeBorImage;
  5235. BorForm: TJclPeBorForm;
  5236. begin
  5237. BorImage := TJclPeBorImage.Create(True);
  5238. try
  5239. BorImage.FileName := FileName;
  5240. Result := BorImage.IsBorlandImage;
  5241. if Result then
  5242. begin
  5243. NamesList.BeginUpdate;
  5244. try
  5245. for I := 0 to BorImage.FormCount - 1 do
  5246. begin
  5247. BorForm := BorImage.Forms[I];
  5248. NamesList.AddObject(BorForm.DisplayName, Pointer(BorForm.ResItem.RawEntryDataSize));
  5249. end;
  5250. finally
  5251. NamesList.EndUpdate;
  5252. end;
  5253. end;
  5254. finally
  5255. BorImage.Free;
  5256. end;
  5257. end;
  5258. function PeBorDependedPackages(const FileName: TFileName; PackagesList: TStrings;
  5259. FullPathName, Descriptions: Boolean): Boolean;
  5260. var
  5261. BorImage: TJclPeBorImage;
  5262. begin
  5263. BorImage := TJclPeBorImage.Create(True);
  5264. try
  5265. BorImage.FileName := FileName;
  5266. Result := BorImage.DependedPackages(PackagesList, FullPathName, Descriptions);
  5267. finally
  5268. BorImage.Free;
  5269. end;
  5270. end;
  5271. {$ENDIF BORLAND}
  5272. // Missing imports checking routines
  5273. function PeFindMissingImports(const FileName: TFileName; MissingImportsList: TStrings): Boolean;
  5274. var
  5275. Cache: TJclPeImagesCache;
  5276. FileImage, LibImage: TJclPeImage;
  5277. L, I: Integer;
  5278. LibItem: TJclPeImportLibItem;
  5279. List: TStringList;
  5280. begin
  5281. Result := False;
  5282. List := nil;
  5283. Cache := TJclPeImagesCache.Create;
  5284. try
  5285. List := TStringList.Create;
  5286. List.Duplicates := dupIgnore;
  5287. List.Sorted := True;
  5288. FileImage := Cache[FileName];
  5289. if FileImage.StatusOK then
  5290. begin
  5291. for L := 0 to FileImage.ImportList.Count - 1 do
  5292. begin
  5293. LibItem := FileImage.ImportList[L];
  5294. LibImage := Cache[LibItem.FileName];
  5295. if LibImage.StatusOK then
  5296. begin
  5297. LibImage.ExportList.PrepareForFastNameSearch;
  5298. for I := 0 to LibItem.Count - 1 do
  5299. if LibImage.ExportList.ItemFromName[LibItem[I].Name] = nil then
  5300. List.Add(LibItem.Name + '=' + LibItem[I].Name);
  5301. end
  5302. else
  5303. List.Add(LibItem.Name + '=');
  5304. end;
  5305. MissingImportsList.Assign(List);
  5306. Result := List.Count > 0;
  5307. end;
  5308. finally
  5309. List.Free;
  5310. Cache.Free;
  5311. end;
  5312. end;
  5313. function PeFindMissingImports(RequiredImportsList, MissingImportsList: TStrings): Boolean;
  5314. var
  5315. Cache: TJclPeImagesCache;
  5316. LibImage: TJclPeImage;
  5317. I, SepPos: Integer;
  5318. List: TStringList;
  5319. S, LibName, ImportName: string;
  5320. begin
  5321. List := nil;
  5322. Cache := TJclPeImagesCache.Create;
  5323. try
  5324. List := TStringList.Create;
  5325. List.Duplicates := dupIgnore;
  5326. List.Sorted := True;
  5327. for I := 0 to RequiredImportsList.Count - 1 do
  5328. begin
  5329. S := RequiredImportsList[I];
  5330. SepPos := Pos('=', S);
  5331. if SepPos = 0 then
  5332. Continue;
  5333. LibName := StrLeft(S, SepPos - 1);
  5334. LibImage := Cache[LibName];
  5335. if LibImage.StatusOK then
  5336. begin
  5337. LibImage.ExportList.PrepareForFastNameSearch;
  5338. ImportName := StrRestOf(S, SepPos + 1);
  5339. if LibImage.ExportList.ItemFromName[ImportName] = nil then
  5340. List.Add(LibName + '=' + ImportName);
  5341. end
  5342. else
  5343. List.Add(LibName + '=');
  5344. end;
  5345. MissingImportsList.Assign(List);
  5346. Result := List.Count > 0;
  5347. finally
  5348. List.Free;
  5349. Cache.Free;
  5350. end;
  5351. end;
  5352. function PeCreateRequiredImportList(const FileName: TFileName; RequiredImportsList: TStrings): Boolean;
  5353. begin
  5354. Result := PeImportedFunctions(FileName, RequiredImportsList, '', True);
  5355. end;
  5356. // Mapped or loaded image related functions
  5357. function PeMapImgNtHeaders32(const BaseAddress: Pointer): PImageNtHeaders32;
  5358. begin
  5359. Result := nil;
  5360. if IsBadReadPtr(BaseAddress, SizeOf(TImageDosHeader)) then
  5361. Exit;
  5362. if (PImageDosHeader(BaseAddress)^.e_magic <> IMAGE_DOS_SIGNATURE) or
  5363. (PImageDosHeader(BaseAddress)^._lfanew = 0) then
  5364. Exit;
  5365. Result := PImageNtHeaders32(TJclAddr(BaseAddress) + DWORD(PImageDosHeader(BaseAddress)^._lfanew));
  5366. if IsBadReadPtr(Result, SizeOf(TImageNtHeaders32)) or
  5367. (Result^.Signature <> IMAGE_NT_SIGNATURE) then
  5368. Result := nil
  5369. end;
  5370. function PeMapImgNtHeaders32(Stream: TStream; const BasePosition: Int64; out NtHeaders32: TImageNtHeaders32): Int64;
  5371. var
  5372. ImageDosHeader: TImageDosHeader;
  5373. begin
  5374. ResetMemory(NtHeaders32, SizeOf(NtHeaders32));
  5375. Result := -1;
  5376. if (Stream.Seek(BasePosition, soBeginning) <> BasePosition) or
  5377. (Stream.Read(ImageDosHeader, SizeOf(ImageDosHeader)) <> SizeOf(ImageDosHeader)) then
  5378. raise EJclPeImageError.CreateRes(@SReadError);
  5379. if (ImageDosHeader.e_magic <> IMAGE_DOS_SIGNATURE) or
  5380. (ImageDosHeader._lfanew = 0) then
  5381. Exit;
  5382. Result := BasePosition + DWORD(ImageDosHeader._lfanew);
  5383. if (Stream.Seek(Result, soBeginning) <> Result) or
  5384. (Stream.Read(NtHeaders32, SizeOf(NtHeaders32)) <> SizeOf(NtHeaders32)) then
  5385. raise EJclPeImageError.CreateRes(@SReadError);
  5386. if NtHeaders32.Signature <> IMAGE_NT_SIGNATURE then
  5387. Result := -1;
  5388. end;
  5389. function PeMapImgNtHeaders64(const BaseAddress: Pointer): PImageNtHeaders64;
  5390. begin
  5391. Result := nil;
  5392. if IsBadReadPtr(BaseAddress, SizeOf(TImageDosHeader)) then
  5393. Exit;
  5394. if (PImageDosHeader(BaseAddress)^.e_magic <> IMAGE_DOS_SIGNATURE) or
  5395. (PImageDosHeader(BaseAddress)^._lfanew = 0) then
  5396. Exit;
  5397. Result := PImageNtHeaders64(TJclAddr(BaseAddress) + DWORD(PImageDosHeader(BaseAddress)^._lfanew));
  5398. if IsBadReadPtr(Result, SizeOf(TImageNtHeaders64)) or
  5399. (Result^.Signature <> IMAGE_NT_SIGNATURE) then
  5400. Result := nil
  5401. end;
  5402. function PeMapImgNtHeaders64(Stream: TStream; const BasePosition: Int64; out NtHeaders64: TImageNtHeaders64): Int64;
  5403. var
  5404. ImageDosHeader: TImageDosHeader;
  5405. begin
  5406. ResetMemory(NtHeaders64, SizeOf(NtHeaders64));
  5407. Result := -1;
  5408. if (Stream.Seek(BasePosition, soBeginning) <> BasePosition) or
  5409. (Stream.Read(ImageDosHeader, SizeOf(ImageDosHeader)) <> SizeOf(ImageDosHeader)) then
  5410. raise EJclPeImageError.CreateRes(@SReadError);
  5411. if (ImageDosHeader.e_magic <> IMAGE_DOS_SIGNATURE) or
  5412. (ImageDosHeader._lfanew = 0) then
  5413. Exit;
  5414. Result := BasePosition + DWORD(ImageDosHeader._lfanew);
  5415. if (Stream.Seek(Result, soBeginning) <> Result) or
  5416. (Stream.Read(NtHeaders64, SizeOf(NtHeaders64)) <> SizeOf(NtHeaders64)) then
  5417. raise EJclPeImageError.CreateRes(@SReadError);
  5418. if NtHeaders64.Signature <> IMAGE_NT_SIGNATURE then
  5419. Result := -1;
  5420. end;
  5421. function PeMapImgSize(const BaseAddress: Pointer): DWORD;
  5422. begin
  5423. case PeMapImgTarget(BaseAddress) of
  5424. taWin32:
  5425. Result := PeMapImgSize32(BaseAddress);
  5426. taWin64:
  5427. Result := PeMapImgSize64(BaseAddress);
  5428. //taUnknown:
  5429. else
  5430. Result := 0;
  5431. end;
  5432. end;
  5433. function PeMapImgSize(Stream: TStream; const BasePosition: Int64): DWORD;
  5434. begin
  5435. case PeMapImgTarget(Stream, BasePosition) of
  5436. taWin32:
  5437. Result := PeMapImgSize32(Stream, BasePosition);
  5438. taWin64:
  5439. Result := PeMapImgSize64(Stream, BasePosition);
  5440. //taUnknown:
  5441. else
  5442. Result := 0;
  5443. end;
  5444. end;
  5445. function PeMapImgSize32(const BaseAddress: Pointer): DWORD;
  5446. var
  5447. NtHeaders32: PImageNtHeaders32;
  5448. begin
  5449. Result := 0;
  5450. NtHeaders32 := PeMapImgNtHeaders32(BaseAddress);
  5451. if Assigned(NtHeaders32) then
  5452. Result := NtHeaders32^.OptionalHeader.SizeOfImage;
  5453. end;
  5454. function PeMapImgSize32(Stream: TStream; const BasePosition: Int64): DWORD;
  5455. var
  5456. NtHeaders32: TImageNtHeaders32;
  5457. begin
  5458. Result := 0;
  5459. if PeMapImgNtHeaders32(Stream, BasePosition, NtHeaders32) <> -1 then
  5460. Result := NtHeaders32.OptionalHeader.SizeOfImage;
  5461. end;
  5462. function PeMapImgSize64(const BaseAddress: Pointer): DWORD;
  5463. var
  5464. NtHeaders64: PImageNtHeaders64;
  5465. begin
  5466. Result := 0;
  5467. NtHeaders64 := PeMapImgNtHeaders64(BaseAddress);
  5468. if Assigned(NtHeaders64) then
  5469. Result := NtHeaders64^.OptionalHeader.SizeOfImage;
  5470. end;
  5471. function PeMapImgSize64(Stream: TStream; const BasePosition: Int64): DWORD;
  5472. var
  5473. NtHeaders64: TImageNtHeaders64;
  5474. begin
  5475. Result := 0;
  5476. if PeMapImgNtHeaders64(Stream, BasePosition, NtHeaders64) <> -1 then
  5477. Result := NtHeaders64.OptionalHeader.SizeOfImage;
  5478. end;
  5479. function PeMapImgLibraryName(const BaseAddress: Pointer): string;
  5480. begin
  5481. case PeMapImgTarget(BaseAddress) of
  5482. taWin32:
  5483. Result := PeMapImgLibraryName32(BaseAddress);
  5484. taWin64:
  5485. Result := PeMapImgLibraryName64(BaseAddress);
  5486. //taUnknown:
  5487. else
  5488. Result := '';
  5489. end;
  5490. end;
  5491. function PeMapImgLibraryName32(const BaseAddress: Pointer): string;
  5492. var
  5493. NtHeaders: PImageNtHeaders32;
  5494. DataDir: TImageDataDirectory;
  5495. ExportDir: PImageExportDirectory;
  5496. UTF8Name: TUTF8String;
  5497. begin
  5498. Result := '';
  5499. NtHeaders := PeMapImgNtHeaders32(BaseAddress);
  5500. if NtHeaders = nil then
  5501. Exit;
  5502. DataDir := NtHeaders^.OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_EXPORT];
  5503. if DataDir.Size = 0 then
  5504. Exit;
  5505. ExportDir := PImageExportDirectory(TJclAddr(BaseAddress) + DataDir.VirtualAddress);
  5506. if IsBadReadPtr(ExportDir, SizeOf(TImageExportDirectory)) or (ExportDir^.Name = 0) then
  5507. Exit;
  5508. UTF8Name := PAnsiChar(TJclAddr(BaseAddress) + ExportDir^.Name);
  5509. if not TryUTF8ToString(UTF8Name, Result) then
  5510. Result := string(UTF8Name);
  5511. end;
  5512. function PeMapImgLibraryName64(const BaseAddress: Pointer): string;
  5513. var
  5514. NtHeaders: PImageNtHeaders64;
  5515. DataDir: TImageDataDirectory;
  5516. ExportDir: PImageExportDirectory;
  5517. UTF8Name: TUTF8String;
  5518. begin
  5519. Result := '';
  5520. NtHeaders := PeMapImgNtHeaders64(BaseAddress);
  5521. if NtHeaders = nil then
  5522. Exit;
  5523. DataDir := NtHeaders^.OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_EXPORT];
  5524. if DataDir.Size = 0 then
  5525. Exit;
  5526. ExportDir := PImageExportDirectory(TJclAddr(BaseAddress) + DataDir.VirtualAddress);
  5527. if IsBadReadPtr(ExportDir, SizeOf(TImageExportDirectory)) or (ExportDir^.Name = 0) then
  5528. Exit;
  5529. UTF8Name := PAnsiChar(TJclAddr(BaseAddress) + ExportDir^.Name);
  5530. if not TryUTF8ToString(UTF8Name, Result) then
  5531. Result := string(UTF8Name);
  5532. end;
  5533. function PeMapImgTarget(const BaseAddress: Pointer): TJclPeTarget;
  5534. var
  5535. ImageNtHeaders: PImageNtHeaders32;
  5536. begin
  5537. Result := taUnknown;
  5538. ImageNtHeaders := PeMapImgNtHeaders32(BaseAddress);
  5539. if Assigned(ImageNtHeaders) then
  5540. case ImageNtHeaders.FileHeader.Machine of
  5541. IMAGE_FILE_MACHINE_I386:
  5542. Result := taWin32;
  5543. IMAGE_FILE_MACHINE_AMD64:
  5544. Result := taWin64;
  5545. end;
  5546. end;
  5547. function PeMapImgTarget(Stream: TStream; const BasePosition: Int64): TJclPeTarget;
  5548. var
  5549. ImageNtHeaders: TImageNtHeaders32;
  5550. begin
  5551. Result := taUnknown;
  5552. if PeMapImgNtHeaders32(Stream, BasePosition, ImageNtHeaders) <> -1 then
  5553. begin
  5554. case ImageNtHeaders.FileHeader.Machine of
  5555. IMAGE_FILE_MACHINE_I386:
  5556. Result := taWin32;
  5557. IMAGE_FILE_MACHINE_AMD64:
  5558. Result := taWin64;
  5559. end;
  5560. end;
  5561. end;
  5562. function PeMapImgSections32(NtHeaders: PImageNtHeaders32): PImageSectionHeader;
  5563. begin
  5564. if NtHeaders = nil then
  5565. Result := nil
  5566. else
  5567. Result := PImageSectionHeader(TJclAddr(@NtHeaders^.OptionalHeader) +
  5568. NtHeaders^.FileHeader.SizeOfOptionalHeader);
  5569. end;
  5570. function PeMapImgSections32(Stream: TStream; const NtHeaders32Position: Int64; const NtHeaders32: TImageNtHeaders32;
  5571. out ImageSectionHeaders: TImageSectionHeaderArray): Int64;
  5572. var
  5573. SectionSize: Integer;
  5574. begin
  5575. if NtHeaders32Position = -1 then
  5576. begin
  5577. SetLength(ImageSectionHeaders, 0);
  5578. Result := -1;
  5579. end
  5580. else
  5581. begin
  5582. SetLength(ImageSectionHeaders, NtHeaders32.FileHeader.NumberOfSections);
  5583. Result := NtHeaders32Position + SizeOf(NtHeaders32.Signature) + SizeOf(NtHeaders32.FileHeader) + NtHeaders32.FileHeader.SizeOfOptionalHeader;
  5584. SectionSize := SizeOf(ImageSectionHeaders[0]) * Length(ImageSectionHeaders);
  5585. if (Stream.Seek(Result, soBeginning) <> Result) or
  5586. (Stream.Read(ImageSectionHeaders[0], SectionSize) <> SectionSize) then
  5587. raise EJclPeImageError.CreateRes(@SReadError);
  5588. end;
  5589. end;
  5590. function PeMapImgSections64(NtHeaders: PImageNtHeaders64): PImageSectionHeader;
  5591. begin
  5592. if NtHeaders = nil then
  5593. Result := nil
  5594. else
  5595. Result := PImageSectionHeader(TJclAddr(@NtHeaders^.OptionalHeader) +
  5596. NtHeaders^.FileHeader.SizeOfOptionalHeader);
  5597. end;
  5598. function PeMapImgSections64(Stream: TStream; const NtHeaders64Position: Int64; const NtHeaders64: TImageNtHeaders64;
  5599. out ImageSectionHeaders: TImageSectionHeaderArray): Int64;
  5600. var
  5601. SectionSize: Integer;
  5602. begin
  5603. if NtHeaders64Position = -1 then
  5604. begin
  5605. SetLength(ImageSectionHeaders, 0);
  5606. Result := -1;
  5607. end
  5608. else
  5609. begin
  5610. SetLength(ImageSectionHeaders, NtHeaders64.FileHeader.NumberOfSections);
  5611. Result := NtHeaders64Position + SizeOf(NtHeaders64.Signature) + SizeOf(NtHeaders64.FileHeader) + NtHeaders64.FileHeader.SizeOfOptionalHeader;
  5612. SectionSize := SizeOf(ImageSectionHeaders[0]) * Length(ImageSectionHeaders);
  5613. if (Stream.Seek(Result, soBeginning) <> Result) or
  5614. (Stream.Read(ImageSectionHeaders[0], SectionSize) <> SectionSize) then
  5615. raise EJclPeImageError.CreateRes(@SReadError);
  5616. end;
  5617. end;
  5618. function PeMapImgFindSection32(NtHeaders: PImageNtHeaders32;
  5619. const SectionName: string): PImageSectionHeader;
  5620. var
  5621. Header: PImageSectionHeader;
  5622. I: Integer;
  5623. P: PAnsiChar;
  5624. UTF8Name: TUTF8String;
  5625. begin
  5626. Result := nil;
  5627. if NtHeaders <> nil then
  5628. begin
  5629. if not TryStringToUTF8(SectionName, UTF8Name) then
  5630. UTF8Name := TUTF8String(SectionName);
  5631. P := PAnsiChar(UTF8Name);
  5632. Header := PeMapImgSections32(NtHeaders);
  5633. for I := 1 to NtHeaders^.FileHeader.NumberOfSections do
  5634. if StrLCompA(PAnsiChar(@Header^.Name), P, IMAGE_SIZEOF_SHORT_NAME) = 0 then
  5635. begin
  5636. Result := Header;
  5637. Break;
  5638. end
  5639. else
  5640. Inc(Header);
  5641. end;
  5642. end;
  5643. function PeMapImgFindSection64(NtHeaders: PImageNtHeaders64;
  5644. const SectionName: string): PImageSectionHeader;
  5645. var
  5646. Header: PImageSectionHeader;
  5647. I: Integer;
  5648. P: PAnsiChar;
  5649. UTF8Name: TUTF8String;
  5650. begin
  5651. Result := nil;
  5652. if NtHeaders <> nil then
  5653. begin
  5654. if not TryStringToUTF8(SectionName, UTF8Name) then
  5655. UTF8Name := TUTF8String(SectionName);
  5656. P := PAnsiChar(UTF8Name);
  5657. Header := PeMapImgSections64(NtHeaders);
  5658. for I := 1 to NtHeaders^.FileHeader.NumberOfSections do
  5659. if StrLCompA(PAnsiChar(@Header^.Name), P, IMAGE_SIZEOF_SHORT_NAME) = 0 then
  5660. begin
  5661. Result := Header;
  5662. Break;
  5663. end
  5664. else
  5665. Inc(Header);
  5666. end;
  5667. end;
  5668. function PeMapImgFindSection(const ImageSectionHeaders: TImageSectionHeaderArray;
  5669. const SectionName: string): SizeInt;
  5670. var
  5671. P: PAnsiChar;
  5672. UTF8Name: TUTF8String;
  5673. begin
  5674. if Length(ImageSectionHeaders) > 0 then
  5675. begin
  5676. if not TryStringToUTF8(SectionName, UTF8Name) then
  5677. UTF8Name := TUTF8String(SectionName);
  5678. P := PAnsiChar(UTF8Name);
  5679. for Result := Low(ImageSectionHeaders) to High(ImageSectionHeaders) do
  5680. if StrLCompA(PAnsiChar(@ImageSectionHeaders[Result].Name), P, IMAGE_SIZEOF_SHORT_NAME) = 0 then
  5681. Exit;
  5682. end;
  5683. Result := -1;
  5684. end;
  5685. function PeMapImgFindSectionFromModule(const BaseAddress: Pointer;
  5686. const SectionName: string): PImageSectionHeader;
  5687. function PeMapImgFindSectionFromModule32(const BaseAddress: Pointer;
  5688. const SectionName: string): PImageSectionHeader;
  5689. var
  5690. NtHeaders32: PImageNtHeaders32;
  5691. begin
  5692. Result := nil;
  5693. NtHeaders32 := PeMapImgNtHeaders32(BaseAddress);
  5694. if Assigned(NtHeaders32) then
  5695. Result := PeMapImgFindSection32(NtHeaders32, SectionName);
  5696. end;
  5697. function PeMapImgFindSectionFromModule64(const BaseAddress: Pointer;
  5698. const SectionName: string): PImageSectionHeader;
  5699. var
  5700. NtHeaders64: PImageNtHeaders64;
  5701. begin
  5702. Result := nil;
  5703. NtHeaders64 := PeMapImgNtHeaders64(BaseAddress);
  5704. if Assigned(NtHeaders64) then
  5705. Result := PeMapImgFindSection64(NtHeaders64, SectionName);
  5706. end;
  5707. begin
  5708. case PeMapImgTarget(BaseAddress) of
  5709. taWin32:
  5710. Result := PeMapImgFindSectionFromModule32(BaseAddress, SectionName);
  5711. taWin64:
  5712. Result := PeMapImgFindSectionFromModule64(BaseAddress, SectionName);
  5713. //taUnknown:
  5714. else
  5715. Result := nil;
  5716. end;
  5717. end;
  5718. function PeMapImgExportedVariables(const Module: HMODULE; const VariablesList: TStrings): Boolean;
  5719. var
  5720. I: Integer;
  5721. begin
  5722. with TJclPeImage.Create(True) do
  5723. try
  5724. AttachLoadedModule(Module);
  5725. Result := StatusOK;
  5726. if Result then
  5727. begin
  5728. VariablesList.BeginUpdate;
  5729. try
  5730. with ExportList do
  5731. for I := 0 to Count - 1 do
  5732. with Items[I] do
  5733. if IsExportedVariable then
  5734. VariablesList.AddObject(Name, MappedAddress);
  5735. finally
  5736. VariablesList.EndUpdate;
  5737. end;
  5738. end;
  5739. finally
  5740. Free;
  5741. end;
  5742. end;
  5743. function PeMapImgResolvePackageThunk(Address: Pointer): Pointer;
  5744. {$IFDEF BORLAND}
  5745. const
  5746. JmpInstructionCode = $25FF;
  5747. type
  5748. PPackageThunk = ^TPackageThunk;
  5749. TPackageThunk = packed record
  5750. JmpInstruction: Word;
  5751. JmpAddress: PPointer;
  5752. end;
  5753. begin
  5754. if not IsCompiledWithPackages then
  5755. Result := Address
  5756. else
  5757. if not IsBadReadPtr(Address, SizeOf(TPackageThunk)) and
  5758. (PPackageThunk(Address)^.JmpInstruction = JmpInstructionCode) then
  5759. Result := PPackageThunk(Address)^.JmpAddress^
  5760. else
  5761. Result := nil;
  5762. end;
  5763. {$ENDIF BORLAND}
  5764. {$IFDEF FPC}
  5765. begin
  5766. Result := Address;
  5767. end;
  5768. {$ENDIF FPC}
  5769. function PeMapFindResource(const Module: HMODULE; const ResourceType: PChar;
  5770. const ResourceName: string): Pointer;
  5771. var
  5772. ResItem: TJclPeResourceItem;
  5773. begin
  5774. Result := nil;
  5775. with TJclPeImage.Create(True) do
  5776. try
  5777. AttachLoadedModule(Module);
  5778. if StatusOK then
  5779. begin
  5780. ResItem := ResourceList.FindResource(ResourceType, PChar(ResourceName));
  5781. if (ResItem <> nil) and ResItem.IsDirectory then
  5782. Result := ResItem.List[0].RawEntryData;
  5783. end;
  5784. finally
  5785. Free;
  5786. end;
  5787. end;
  5788. //=== { TJclPeSectionStream } ================================================
  5789. constructor TJclPeSectionStream.Create(Instance: HMODULE; const ASectionName: string);
  5790. begin
  5791. inherited Create;
  5792. Initialize(Instance, ASectionName);
  5793. end;
  5794. procedure TJclPeSectionStream.Initialize(Instance: HMODULE; const ASectionName: string);
  5795. var
  5796. Header: PImageSectionHeader;
  5797. NtHeaders32: PImageNtHeaders32;
  5798. NtHeaders64: PImageNtHeaders64;
  5799. DataSize: Integer;
  5800. begin
  5801. FInstance := Instance;
  5802. case PeMapImgTarget(Pointer(Instance)) of
  5803. taWin32:
  5804. begin
  5805. NtHeaders32 := PeMapImgNtHeaders32(Pointer(Instance));
  5806. if NtHeaders32 = nil then
  5807. raise EJclPeImageError.CreateRes(@RsPeNotPE);
  5808. Header := PeMapImgFindSection32(NtHeaders32, ASectionName);
  5809. end;
  5810. taWin64:
  5811. begin
  5812. NtHeaders64 := PeMapImgNtHeaders64(Pointer(Instance));
  5813. if NtHeaders64 = nil then
  5814. raise EJclPeImageError.CreateRes(@RsPeNotPE);
  5815. Header := PeMapImgFindSection64(NtHeaders64, ASectionName);
  5816. end;
  5817. //toUnknown:
  5818. else
  5819. raise EJclPeImageError.CreateRes(@RsPeUnknownTarget);
  5820. end;
  5821. if Header = nil then
  5822. raise EJclPeImageError.CreateResFmt(@RsPeSectionNotFound, [ASectionName]);
  5823. // Borland and Microsoft seems to have swapped the meaning of this items.
  5824. DataSize := Min(Header^.SizeOfRawData, Header^.Misc.VirtualSize);
  5825. SetPointer(Pointer(FInstance + Header^.VirtualAddress), DataSize);
  5826. FSectionHeader := Header^;
  5827. end;
  5828. function TJclPeSectionStream.Write(const Buffer; Count: Integer): Longint;
  5829. begin
  5830. raise EJclPeImageError.CreateRes(@RsPeReadOnlyStream);
  5831. end;
  5832. //=== { TJclPeMapImgHookItem } ===============================================
  5833. constructor TJclPeMapImgHookItem.Create(AList: TObjectList;
  5834. const AFunctionName: string; const AModuleName: string;
  5835. ABaseAddress, ANewAddress, AOriginalAddress: Pointer);
  5836. begin
  5837. inherited Create;
  5838. FList := AList;
  5839. FFunctionName := AFunctionName;
  5840. FModuleName := AModuleName;
  5841. FBaseAddress := ABaseAddress;
  5842. FNewAddress := ANewAddress;
  5843. FOriginalAddress := AOriginalAddress;
  5844. end;
  5845. destructor TJclPeMapImgHookItem.Destroy;
  5846. begin
  5847. if FBaseAddress <> nil then
  5848. InternalUnhook;
  5849. inherited Destroy;
  5850. end;
  5851. function TJclPeMapImgHookItem.InternalUnhook: Boolean;
  5852. var
  5853. Buf: TMemoryBasicInformation;
  5854. begin
  5855. Buf.AllocationBase := nil;
  5856. if (VirtualQuery(FBaseAddress, Buf, SizeOf(Buf)) = SizeOf(Buf)) and (Buf.State and MEM_FREE = 0) then
  5857. Result := TJclPeMapImgHooks.ReplaceImport(FBaseAddress, ModuleName, NewAddress, OriginalAddress)
  5858. else
  5859. Result := True; // PE image is not available anymore (DLL got unloaded)
  5860. if Result then
  5861. FBaseAddress := nil;
  5862. end;
  5863. function TJclPeMapImgHookItem.Unhook: Boolean;
  5864. begin
  5865. Result := InternalUnhook;
  5866. if Result then
  5867. FList.Remove(Self);
  5868. end;
  5869. //=== { TJclPeMapImgHooks } ==================================================
  5870. type
  5871. PWin9xDebugThunk32 = ^TWin9xDebugThunk32;
  5872. TWin9xDebugThunk32 = packed record
  5873. PUSH: Byte; // PUSH instruction opcode ($68)
  5874. Addr: DWORD; // The actual address of the DLL routine
  5875. JMP: Byte; // JMP instruction opcode ($E9)
  5876. Rel: DWORD; // Relative displacement (a Kernel32 address)
  5877. end;
  5878. function TJclPeMapImgHooks.GetItemFromNewAddress(NewAddress: Pointer): TJclPeMapImgHookItem;
  5879. var
  5880. I: Integer;
  5881. begin
  5882. Result := nil;
  5883. for I := 0 to Count - 1 do
  5884. if Items[I].NewAddress = NewAddress then
  5885. begin
  5886. Result := Items[I];
  5887. Break;
  5888. end;
  5889. end;
  5890. function TJclPeMapImgHooks.GetItemFromOriginalAddress(OriginalAddress: Pointer): TJclPeMapImgHookItem;
  5891. var
  5892. I: Integer;
  5893. begin
  5894. Result := nil;
  5895. for I := 0 to Count - 1 do
  5896. if Items[I].OriginalAddress = OriginalAddress then
  5897. begin
  5898. Result := Items[I];
  5899. Break;
  5900. end;
  5901. end;
  5902. function TJclPeMapImgHooks.GetItems(Index: Integer): TJclPeMapImgHookItem;
  5903. begin
  5904. Result := TJclPeMapImgHookItem(Get(Index));
  5905. end;
  5906. function TJclPeMapImgHooks.HookImport(Base: Pointer; const ModuleName: string;
  5907. const FunctionName: string; NewAddress: Pointer; var OriginalAddress: Pointer): Boolean;
  5908. var
  5909. ModuleHandle: THandle;
  5910. OriginalItem: TJclPeMapImgHookItem;
  5911. UTF8Name: TUTF8String;
  5912. begin
  5913. ModuleHandle := GetModuleHandle(PChar(ModuleName));
  5914. Result := (ModuleHandle <> 0);
  5915. if not Result then
  5916. begin
  5917. SetLastError(ERROR_MOD_NOT_FOUND);
  5918. Exit;
  5919. end;
  5920. if not TryStringToUTF8(FunctionName, UTF8Name) then
  5921. UTF8Name := TUTF8String(FunctionName);
  5922. OriginalAddress := GetProcAddress(ModuleHandle, PAnsiChar(UTF8Name));
  5923. Result := (OriginalAddress <> nil);
  5924. if not Result then
  5925. begin
  5926. SetLastError(ERROR_PROC_NOT_FOUND);
  5927. Exit;
  5928. end;
  5929. OriginalItem := ItemFromOriginalAddress[OriginalAddress];
  5930. Result := ((OriginalItem = nil) or (OriginalItem.ModuleName = ModuleName)) and
  5931. (NewAddress <> nil) and (OriginalAddress <> NewAddress);
  5932. if not Result then
  5933. begin
  5934. SetLastError(ERROR_ALREADY_EXISTS);
  5935. Exit;
  5936. end;
  5937. if Result then
  5938. Result := ReplaceImport(Base, ModuleName, OriginalAddress, NewAddress);
  5939. if Result then
  5940. begin
  5941. Add(TJclPeMapImgHookItem.Create(Self, FunctionName, ModuleName, Base,
  5942. NewAddress, OriginalAddress));
  5943. end
  5944. else
  5945. SetLastError(ERROR_INVALID_PARAMETER);
  5946. end;
  5947. class function TJclPeMapImgHooks.IsWin9xDebugThunk(P: Pointer): Boolean;
  5948. begin
  5949. with PWin9xDebugThunk32(P)^ do
  5950. Result := (PUSH = $68) and (JMP = $E9);
  5951. end;
  5952. class function TJclPeMapImgHooks.ReplaceImport(Base: Pointer; const ModuleName: string;
  5953. FromProc, ToProc: Pointer): Boolean;
  5954. var
  5955. {$IFDEF CPU32}
  5956. FromProcDebugThunk32, ImportThunk32: PWin9xDebugThunk32;
  5957. IsThunked: Boolean;
  5958. NtHeader: PImageNtHeaders32;
  5959. ImportEntry: PImageThunkData32;
  5960. {$ENDIF CPU32}
  5961. {$IFDEF CPU64}
  5962. NtHeader: PImageNtHeaders64;
  5963. ImportEntry: PImageThunkData64;
  5964. {$ENDIF CPU64}
  5965. ImportDir: TImageDataDirectory;
  5966. ImportDesc: PImageImportDescriptor;
  5967. CurrName, RefName: PAnsiChar;
  5968. FoundProc: Boolean;
  5969. WrittenBytes: Cardinal;
  5970. UTF8Name: TUTF8String;
  5971. begin
  5972. Result := False;
  5973. {$IFDEF CPU32}
  5974. FromProcDebugThunk32 := PWin9xDebugThunk32(FromProc);
  5975. IsThunked := (Win32Platform <> VER_PLATFORM_WIN32_NT) and IsWin9xDebugThunk(FromProcDebugThunk32);
  5976. NtHeader := PeMapImgNtHeaders32(Base);
  5977. {$ENDIF CPU32}
  5978. {$IFDEF CPU64}
  5979. NtHeader := PeMapImgNtHeaders64(Base);
  5980. {$ENDIF CPU64}
  5981. if NtHeader = nil then
  5982. Exit;
  5983. ImportDir := NtHeader.OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT];
  5984. if ImportDir.VirtualAddress = 0 then
  5985. Exit;
  5986. ImportDesc := PImageImportDescriptor(TJclAddr(Base) + ImportDir.VirtualAddress);
  5987. if not TryStringToUTF8(ModuleName, UTF8Name) then
  5988. UTF8Name := TUTF8String(ModuleName);
  5989. RefName := PAnsiChar(UTF8Name);
  5990. while ImportDesc^.Name <> 0 do
  5991. begin
  5992. CurrName := PAnsiChar(Base) + ImportDesc^.Name;
  5993. if StrICompA(CurrName, RefName) = 0 then
  5994. begin
  5995. {$IFDEF CPU32}
  5996. ImportEntry := PImageThunkData32(TJclAddr(Base) + ImportDesc^.FirstThunk);
  5997. {$ENDIF CPU32}
  5998. {$IFDEF CPU64}
  5999. ImportEntry := PImageThunkData64(TJclAddr(Base) + ImportDesc^.FirstThunk);
  6000. {$ENDIF CPU64}
  6001. while ImportEntry^.Function_ <> 0 do
  6002. begin
  6003. {$IFDEF CPU32}
  6004. if IsThunked then
  6005. begin
  6006. ImportThunk32 := PWin9xDebugThunk32(ImportEntry^.Function_);
  6007. FoundProc := IsWin9xDebugThunk(ImportThunk32) and (ImportThunk32^.Addr = FromProcDebugThunk32^.Addr);
  6008. end
  6009. else
  6010. {$ENDIF CPU32}
  6011. FoundProc := Pointer(ImportEntry^.Function_) = FromProc;
  6012. if FoundProc then
  6013. Result := WriteProtectedMemory(@ImportEntry^.Function_, @ToProc, SizeOf(ToProc), WrittenBytes);
  6014. Inc(ImportEntry);
  6015. end;
  6016. end;
  6017. Inc(ImportDesc);
  6018. end;
  6019. end;
  6020. class function TJclPeMapImgHooks.SystemBase: Pointer;
  6021. begin
  6022. Result := Pointer(SystemTObjectInstance);
  6023. end;
  6024. procedure TJclPeMapImgHooks.UnhookAll;
  6025. var
  6026. I: Integer;
  6027. begin
  6028. I := 0;
  6029. while I < Count do
  6030. if not Items[I].Unhook then
  6031. Inc(I);
  6032. end;
  6033. function TJclPeMapImgHooks.UnhookByNewAddress(NewAddress: Pointer): Boolean;
  6034. var
  6035. Item: TJclPeMapImgHookItem;
  6036. begin
  6037. Item := ItemFromNewAddress[NewAddress];
  6038. Result := (Item <> nil) and Item.Unhook;
  6039. end;
  6040. procedure TJclPeMapImgHooks.UnhookByBaseAddress(BaseAddress: Pointer);
  6041. var
  6042. I: Integer;
  6043. begin
  6044. for I := Count - 1 downto 0 do
  6045. if Items[I].BaseAddress = BaseAddress then
  6046. Items[I].Unhook;
  6047. end;
  6048. // Image access under a debbuger
  6049. {$IFDEF USE_64BIT_TYPES}
  6050. function InternalReadProcMem(ProcessHandle: THandle; Address: DWORD;
  6051. Buffer: Pointer; Size: SIZE_T): Boolean;
  6052. var
  6053. BR: SIZE_T;
  6054. {$ELSE}
  6055. function InternalReadProcMem(ProcessHandle: THandle; Address: DWORD;
  6056. Buffer: Pointer; Size: Integer): Boolean;
  6057. var
  6058. BR: DWORD;
  6059. {$ENDIF}
  6060. begin
  6061. BR := 0;
  6062. Result := ReadProcessMemory(ProcessHandle, Pointer(Address), Buffer, Size, BR);
  6063. end;
  6064. // TODO: 64 bit version
  6065. function PeDbgImgNtHeaders32(ProcessHandle: THandle; BaseAddress: TJclAddr32;
  6066. var NtHeaders: TImageNtHeaders32): Boolean;
  6067. var
  6068. DosHeader: TImageDosHeader;
  6069. begin
  6070. Result := False;
  6071. ResetMemory(NtHeaders, SizeOf(NtHeaders));
  6072. ResetMemory(DosHeader, SizeOf(DosHeader));
  6073. if not InternalReadProcMem(ProcessHandle, TJclAddr32(BaseAddress), @DosHeader, SizeOf(DosHeader)) then
  6074. Exit;
  6075. if DosHeader.e_magic <> IMAGE_DOS_SIGNATURE then
  6076. Exit;
  6077. Result := InternalReadProcMem(ProcessHandle, TJclAddr32(BaseAddress) + TJclAddr32(DosHeader._lfanew),
  6078. @NtHeaders, SizeOf(TImageNtHeaders32));
  6079. end;
  6080. // TODO: 64 bit version
  6081. function PeDbgImgLibraryName32(ProcessHandle: THandle; BaseAddress: TJclAddr32;
  6082. var Name: string): Boolean;
  6083. var
  6084. NtHeaders32: TImageNtHeaders32;
  6085. DataDir: TImageDataDirectory;
  6086. ExportDir: TImageExportDirectory;
  6087. UTF8Name: TUTF8String;
  6088. begin
  6089. Name := '';
  6090. NtHeaders32.Signature := 0;
  6091. Result := PeDbgImgNtHeaders32(ProcessHandle, BaseAddress, NtHeaders32);
  6092. if not Result then
  6093. Exit;
  6094. DataDir := NtHeaders32.OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_EXPORT];
  6095. if DataDir.Size = 0 then
  6096. Exit;
  6097. if not InternalReadProcMem(ProcessHandle, TJclAddr(BaseAddress) + DataDir.VirtualAddress,
  6098. @ExportDir, SizeOf(ExportDir)) then
  6099. Exit;
  6100. if ExportDir.Name = 0 then
  6101. Exit;
  6102. SetLength(UTF8Name, MAX_PATH);
  6103. if InternalReadProcMem(ProcessHandle, TJclAddr(BaseAddress) + ExportDir.Name, PAnsiChar(UTF8Name), MAX_PATH) then
  6104. begin
  6105. StrResetLength(UTF8Name);
  6106. if not TryUTF8ToString(UTF8Name, Name) then
  6107. Name := string(UTF8Name);
  6108. end
  6109. else
  6110. Name := '';
  6111. end;
  6112. // Borland BPL packages name unmangling
  6113. function PeBorUnmangleName(const Name: string; out Unmangled: string;
  6114. out Description: TJclBorUmDescription; out BasePos: Integer): TJclBorUmResult;
  6115. var
  6116. NameP, NameU, NameUFirst: PAnsiChar;
  6117. QualifierFound, LinkProcFound: Boolean;
  6118. UTF8Unmangled, UTF8Name: TUTF8String;
  6119. procedure MarkQualifier;
  6120. begin
  6121. if not QualifierFound then
  6122. begin
  6123. QualifierFound := True;
  6124. BasePos := NameU - NameUFirst + 2;
  6125. end;
  6126. end;
  6127. procedure ReadSpecialSymbol;
  6128. var
  6129. SymbolLength: Integer;
  6130. begin
  6131. SymbolLength := 0;
  6132. while CharIsDigit(Char(NameP^)) do
  6133. begin
  6134. SymbolLength := SymbolLength * 10 + Ord(NameP^) - 48;
  6135. Inc(NameP);
  6136. end;
  6137. while (SymbolLength > 0) and (NameP^ <> #0) do
  6138. begin
  6139. if NameP^ = '@' then
  6140. begin
  6141. MarkQualifier;
  6142. NameU^ := '.';
  6143. end
  6144. else
  6145. NameU^ := NameP^;
  6146. Inc(NameP);
  6147. Inc(NameU);
  6148. Dec(SymbolLength);
  6149. end;
  6150. end;
  6151. procedure ReadRTTI;
  6152. begin
  6153. if StrLCompA(NameP, '$xp$', 4) = 0 then
  6154. begin
  6155. Inc(NameP, 4);
  6156. Description.Kind := skRTTI;
  6157. QualifierFound := False;
  6158. ReadSpecialSymbol;
  6159. if QualifierFound then
  6160. Include(Description.Modifiers, smQualified);
  6161. end
  6162. else
  6163. Result := urError;
  6164. end;
  6165. procedure ReadNameSymbol;
  6166. begin
  6167. if NameP^ = '@' then
  6168. begin
  6169. LinkProcFound := True;
  6170. Inc(NameP);
  6171. end;
  6172. while CharIsValidIdentifierLetter(Char(NameP^)) do
  6173. begin
  6174. NameU^ := NameP^;
  6175. Inc(NameP);
  6176. Inc(NameU);
  6177. end;
  6178. end;
  6179. procedure ReadName;
  6180. begin
  6181. Description.Kind := skData;
  6182. QualifierFound := False;
  6183. LinkProcFound := False;
  6184. repeat
  6185. ReadNameSymbol;
  6186. if LinkProcFound and not QualifierFound then
  6187. LinkProcFound := False;
  6188. case NameP^ of
  6189. '@':
  6190. case (NameP + 1)^ of
  6191. #0:
  6192. begin
  6193. Description.Kind := skVTable;
  6194. Break;
  6195. end;
  6196. '$':
  6197. begin
  6198. if (NameP + 2)^ = 'b' then
  6199. begin
  6200. case (NameP + 3)^ of
  6201. 'c':
  6202. Description.Kind := skConstructor;
  6203. 'd':
  6204. Description.Kind := skDestructor;
  6205. end;
  6206. Inc(NameP, 6);
  6207. end
  6208. else
  6209. Description.Kind := skFunction;
  6210. Break; // no parameters unmangling yet
  6211. end;
  6212. else
  6213. MarkQualifier;
  6214. NameU^ := '.';
  6215. Inc(NameU);
  6216. Inc(NameP);
  6217. end;
  6218. '$':
  6219. begin
  6220. Description.Kind := skFunction;
  6221. Break; // no parameters unmangling yet
  6222. end;
  6223. else
  6224. Break;
  6225. end;
  6226. until False;
  6227. if QualifierFound then
  6228. Include(Description.Modifiers, smQualified);
  6229. if LinkProcFound then
  6230. Include(Description.Modifiers, smLinkProc);
  6231. end;
  6232. begin
  6233. if not TryStringToUTF8(Name, UTF8Name) then
  6234. UTF8Name := TUTF8String(Name);
  6235. NameP := PAnsiChar(UTF8Name);
  6236. Result := urError;
  6237. case NameP^ of
  6238. '@':
  6239. Result := urOk;
  6240. '?':
  6241. Result := urMicrosoft;
  6242. '_', 'A'..'Z', 'a'..'z':
  6243. Result := urNotMangled;
  6244. end;
  6245. if Result <> urOk then
  6246. Exit;
  6247. Inc(NameP);
  6248. SetLength(UTF8UnMangled, 1024);
  6249. NameU := PAnsiChar(UTF8UnMangled);
  6250. NameUFirst := NameU;
  6251. Description.Modifiers := [];
  6252. BasePos := 1;
  6253. case NameP^ of
  6254. '$':
  6255. ReadRTTI;
  6256. '_', 'A'..'Z', 'a'..'z':
  6257. ReadName;
  6258. else
  6259. Result := urError;
  6260. end;
  6261. NameU^ := #0;
  6262. SetLength(UTF8Unmangled, StrLenA(PAnsiChar(UTF8Unmangled))); // SysUtils prefix due to compiler bug
  6263. if not TryUTF8ToString(UTF8Unmangled, Unmangled) then
  6264. Unmangled := string(UTF8Unmangled);
  6265. end;
  6266. function PeBorUnmangleName(const Name: string; out Unmangled: string;
  6267. out Description: TJclBorUmDescription): TJclBorUmResult;
  6268. var
  6269. BasePos: Integer;
  6270. begin
  6271. Result := PeBorUnmangleName(Name, Unmangled, Description, BasePos);
  6272. end;
  6273. function PeBorUnmangleName(const Name: string; out Unmangled: string): TJclBorUmResult;
  6274. var
  6275. Description: TJclBorUmDescription;
  6276. BasePos: Integer;
  6277. begin
  6278. Result := PeBorUnmangleName(Name, Unmangled, Description, BasePos);
  6279. end;
  6280. function PeBorUnmangleName(const Name: string): string;
  6281. var
  6282. Unmangled: string;
  6283. Description: TJclBorUmDescription;
  6284. BasePos: Integer;
  6285. begin
  6286. if PeBorUnmangleName(Name, Unmangled, Description, BasePos) = urOk then
  6287. Result := Unmangled
  6288. else
  6289. Result := '';
  6290. end;
  6291. function PeIsNameMangled(const Name: string): TJclPeUmResult;
  6292. begin
  6293. Result := umNotMangled;
  6294. if Length(Name) > 0 then
  6295. case Name[1] of
  6296. '@':
  6297. Result := umBorland;
  6298. '?':
  6299. Result := umMicrosoft;
  6300. end;
  6301. end;
  6302. type
  6303. TUndecorateSymbolNameA = function (DecoratedName: PAnsiChar;
  6304. UnDecoratedName: PAnsiChar; UndecoratedLength: DWORD; Flags: DWORD): DWORD; stdcall;
  6305. // 'imagehlp.dll' 'UnDecorateSymbolName'
  6306. TUndecorateSymbolNameW = function (DecoratedName: PWideChar;
  6307. UnDecoratedName: PWideChar; UndecoratedLength: DWORD; Flags: DWORD): DWORD; stdcall;
  6308. // 'imagehlp.dll' 'UnDecorateSymbolNameW'
  6309. var
  6310. UndecorateSymbolNameA: TUndecorateSymbolNameA = nil;
  6311. UndecorateSymbolNameAFailed: Boolean = False;
  6312. UndecorateSymbolNameW: TUndecorateSymbolNameW = nil;
  6313. UndecorateSymbolNameWFailed: Boolean = False;
  6314. function UndecorateSymbolName(const DecoratedName: string; out UnMangled: string; Flags: DWORD): Boolean;
  6315. const
  6316. ModuleName = 'imagehlp.dll';
  6317. BufferSize = 512;
  6318. var
  6319. ModuleHandle: HMODULE;
  6320. WideBuffer: WideString;
  6321. AnsiBuffer: AnsiString;
  6322. Res: DWORD;
  6323. begin
  6324. Result := False;
  6325. if ((not Assigned(UndecorateSymbolNameA)) and (not UndecorateSymbolNameAFailed)) or
  6326. ((not Assigned(UndecorateSymbolNameW)) and (not UndecorateSymbolNameWFailed)) then
  6327. begin
  6328. ModuleHandle := GetModuleHandle(ModuleName);
  6329. if ModuleHandle = 0 then
  6330. begin
  6331. ModuleHandle := SafeLoadLibrary(ModuleName);
  6332. if ModuleHandle = 0 then
  6333. Exit;
  6334. end;
  6335. UndecorateSymbolNameA := GetProcAddress(ModuleHandle, 'UnDecorateSymbolName');
  6336. UndecorateSymbolNameAFailed := not Assigned(UndecorateSymbolNameA);
  6337. UndecorateSymbolNameW := GetProcAddress(ModuleHandle, 'UnDecorateSymbolNameW');
  6338. UndecorateSymbolNameWFailed := not Assigned(UndecorateSymbolNameW);
  6339. end;
  6340. if Assigned(UndecorateSymbolNameW) then
  6341. begin
  6342. SetLength(WideBuffer, BufferSize);
  6343. Res := UnDecorateSymbolNameW(PWideChar({$IFNDEF UNICODE}WideString{$ENDIF}(DecoratedName)), PWideChar(WideBuffer), BufferSize, Flags);
  6344. if Res > 0 then
  6345. begin
  6346. StrResetLength(WideBuffer);
  6347. UnMangled := string(WideBuffer);
  6348. Result := True;
  6349. end;
  6350. end
  6351. else
  6352. if Assigned(UndecorateSymbolNameA) then
  6353. begin
  6354. SetLength(AnsiBuffer, BufferSize);
  6355. Res := UnDecorateSymbolNameA(PAnsiChar(AnsiString(DecoratedName)), PAnsiChar(AnsiBuffer), BufferSize, Flags);
  6356. if Res > 0 then
  6357. begin
  6358. StrResetLength(AnsiBuffer);
  6359. UnMangled := string(AnsiBuffer);
  6360. Result := True;
  6361. end;
  6362. end;
  6363. // For some functions UnDecorateSymbolName returns 'long'
  6364. if Result and (UnMangled = 'long') then
  6365. UnMangled := DecoratedName;
  6366. end;
  6367. function PeUnmangleName(const Name: string; out Unmangled: string): TJclPeUmResult;
  6368. begin
  6369. Result := umNotMangled;
  6370. case PeBorUnmangleName(Name, Unmangled) of
  6371. urOk:
  6372. Result := umBorland;
  6373. urMicrosoft:
  6374. if UndecorateSymbolName(Name, Unmangled, UNDNAME_NAME_ONLY) then
  6375. Result := umMicrosoft;
  6376. end;
  6377. if Result = umNotMangled then
  6378. Unmangled := Name;
  6379. end;
  6380. {$IFDEF UNITVERSIONING}
  6381. initialization
  6382. RegisterUnitVersion(HInstance, UnitVersioning);
  6383. finalization
  6384. UnregisterUnitVersion(HInstance);
  6385. {$ENDIF UNITVERSIONING}
  6386. end.