JclPeImage.pas 224 KB

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