exeinfo.pp 30 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178
  1. {
  2. This file is part of the Free Pascal run time library.
  3. Copyright (c) 2008 by Peter Vreman
  4. Executable file reading functions
  5. See the file COPYING.FPC, included in this distribution,
  6. for details about the copyright.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  10. **********************************************************************}
  11. {
  12. This unit should not be compiled in objfpc mode, since this would make it
  13. dependent on objpas unit.
  14. }
  15. unit exeinfo;
  16. interface
  17. {$S-}
  18. type
  19. TExeFile=record
  20. f : file;
  21. // cached filesize
  22. size : int64;
  23. isopen : boolean;
  24. nsects : longint;
  25. sechdrofs,
  26. secstrofs : ptruint;
  27. processaddress : ptruint;
  28. FunctionRelative: boolean;
  29. // Offset of the binary image forming permanent offset to all retrieved values
  30. ImgOffset: ptruint;
  31. filename : string;
  32. // Allocate static buffer for reading data
  33. buf : array[0..4095] of byte;
  34. bufsize,
  35. bufcnt : longint;
  36. end;
  37. function OpenExeFile(var e:TExeFile;const fn:string):boolean;
  38. function FindExeSection(var e:TExeFile;const secname:string;var secofs,seclen:longint):boolean;
  39. function CloseExeFile(var e:TExeFile):boolean;
  40. function ReadDebugLink(var e:TExeFile;var dbgfn:string):boolean;
  41. procedure GetModuleByAddr(addr: pointer; var baseaddr: pointer; var filename: string);
  42. implementation
  43. uses
  44. strings{$ifdef windows},windows{$endif windows};
  45. {$ifdef unix}
  46. procedure GetModuleByAddr(addr: pointer; var baseaddr: pointer; var filename: string);
  47. begin
  48. if assigned(UnixGetModuleByAddrHook) then
  49. UnixGetModuleByAddrHook(addr,baseaddr,filename)
  50. else
  51. begin
  52. baseaddr:=nil;
  53. filename:=ParamStr(0);
  54. end;
  55. end;
  56. {$else unix}
  57. {$ifdef windows}
  58. var
  59. Tmm: TMemoryBasicInformation;
  60. {$ifdef wince}
  61. TST: array[0..Max_Path] of WideChar;
  62. {$else wince}
  63. TST: array[0..Max_Path] of Char;
  64. {$endif wince}
  65. procedure GetModuleByAddr(addr: pointer; var baseaddr: pointer; var filename: string);
  66. begin
  67. baseaddr:=nil;
  68. if VirtualQuery(addr, @Tmm, SizeOf(Tmm))<>sizeof(Tmm) then
  69. filename:=ParamStr(0)
  70. else
  71. begin
  72. baseaddr:=Tmm.AllocationBase;
  73. TST[0]:= #0;
  74. GetModuleFileName(THandle(Tmm.AllocationBase), TST, Length(TST));
  75. {$ifdef wince}
  76. filename:= String(PWideChar(@TST));
  77. {$else wince}
  78. filename:= String(PChar(@TST));
  79. {$endif wince}
  80. end;
  81. end;
  82. {$else windows}
  83. procedure GetModuleByAddr(addr: pointer; var baseaddr: pointer; var filename: string);
  84. begin
  85. baseaddr:= nil;
  86. filename:=ParamStr(0);
  87. end;
  88. {$endif windows}
  89. {$endif unix}
  90. {****************************************************************************
  91. Executable Loaders
  92. ****************************************************************************}
  93. {$if defined(netbsd) or defined(freebsd) or defined(linux) or defined(sunos)}
  94. {$ifdef cpu64}
  95. {$define ELF64}
  96. {$else}
  97. {$define ELF32}
  98. {$endif}
  99. {$endif}
  100. {$if defined(win32) or defined(wince)}
  101. {$define PE32}
  102. {$endif}
  103. {$if defined(win64)}
  104. {$define PE32PLUS}
  105. {$endif}
  106. {$ifdef netwlibc}
  107. {$define netware}
  108. {$endif}
  109. {$IFDEF OS2}
  110. {$DEFINE EMX}
  111. {$ENDIF OS2}
  112. {****************************************************************************
  113. DOS Stub
  114. ****************************************************************************}
  115. {$if defined(EMX) or defined(PE32) or defined(PE32PLUS) or defined(GO32V2)}
  116. type
  117. tdosheader = packed record
  118. e_magic : word;
  119. e_cblp : word;
  120. e_cp : word;
  121. e_crlc : word;
  122. e_cparhdr : word;
  123. e_minalloc : word;
  124. e_maxalloc : word;
  125. e_ss : word;
  126. e_sp : word;
  127. e_csum : word;
  128. e_ip : word;
  129. e_cs : word;
  130. e_lfarlc : word;
  131. e_ovno : word;
  132. e_res : array[0..3] of word;
  133. e_oemid : word;
  134. e_oeminfo : word;
  135. e_res2 : array[0..9] of word;
  136. e_lfanew : longint;
  137. end;
  138. {$endif EMX or PE32 or PE32PLUS or GO32v2}
  139. {****************************************************************************
  140. NLM
  141. ****************************************************************************}
  142. {$ifdef netware}
  143. function getByte(var f:file):byte;
  144. begin
  145. BlockRead (f,getByte,1);
  146. end;
  147. procedure Skip (var f:file; bytes : longint);
  148. var i : longint;
  149. begin
  150. for i := 1 to bytes do getbyte(f);
  151. end;
  152. function get0String (var f:file) : string;
  153. var c : char;
  154. begin
  155. get0String := '';
  156. c := char (getbyte(f));
  157. while (c <> #0) do
  158. begin
  159. get0String := get0String + c;
  160. c := char (getbyte(f));
  161. end;
  162. end;
  163. function getint32 (var f:file): longint;
  164. begin
  165. blockread (F, getint32, 4);
  166. end;
  167. const SIZE_OF_NLM_INTERNAL_FIXED_HEADER = 130;
  168. SIZE_OF_NLM_INTERNAL_VERSION_HEADER = 32;
  169. SIZE_OF_NLM_INTERNAL_EXTENDED_HEADER = 124;
  170. function openNetwareNLM(var e:TExeFile):boolean;
  171. var valid : boolean;
  172. name : string;
  173. hdrLength,
  174. dataOffset,
  175. dataLength : longint;
  176. function getLString : String;
  177. var Res:string;
  178. begin
  179. blockread (e.F, res, 1);
  180. if length (res) > 0 THEN
  181. blockread (e.F, res[1], length (res));
  182. getbyte(e.f);
  183. getLString := res;
  184. end;
  185. function getFixString (Len : byte) : string;
  186. var i : byte;
  187. begin
  188. getFixString := '';
  189. for I := 1 to Len do
  190. getFixString := getFixString + char (getbyte(e.f));
  191. end;
  192. function getword : word;
  193. begin
  194. blockread (e.F, getword, 2);
  195. end;
  196. begin
  197. e.sechdrofs := 0;
  198. openNetwareNLM:=false;
  199. // read and check header
  200. Skip (e.f,SIZE_OF_NLM_INTERNAL_FIXED_HEADER);
  201. getLString; // NLM Description
  202. getInt32(e.f); // Stacksize
  203. getInt32(e.f); // Reserved
  204. skip(e.f,5); // old Thread Name
  205. getLString; // Screen Name
  206. getLString; // Thread Name
  207. hdrLength := -1;
  208. dataOffset := -1;
  209. dataLength := -1;
  210. valid := true;
  211. repeat
  212. name := getFixString (8);
  213. if (name = 'VeRsIoN#') then
  214. begin
  215. Skip (e.f,SIZE_OF_NLM_INTERNAL_VERSION_HEADER-8);
  216. end else
  217. if (name = 'CoPyRiGh') then
  218. begin
  219. getword; // T=
  220. getLString; // Copyright String
  221. end else
  222. if (name = 'MeSsAgEs') then
  223. begin
  224. skip (e.f,SIZE_OF_NLM_INTERNAL_EXTENDED_HEADER - 8);
  225. end else
  226. if (name = 'CuStHeAd') then
  227. begin
  228. hdrLength := getInt32(e.f);
  229. dataOffset := getInt32(e.f);
  230. dataLength := getInt32(e.f);
  231. Skip (e.f,8); // dateStamp
  232. Valid := false;
  233. end else
  234. Valid := false;
  235. until not valid;
  236. if (hdrLength = -1) or (dataOffset = -1) or (dataLength = -1) then
  237. exit;
  238. Seek (e.F, dataOffset);
  239. e.sechdrofs := dataOffset;
  240. openNetwareNLM := (e.sechdrofs > 0);
  241. end;
  242. function FindSectionNetwareNLM(var e:TExeFile;const asecname:string;var secofs,seclen:longint):boolean;
  243. var name : string;
  244. alignAmount : longint;
  245. begin
  246. seek(e.f,e.sechdrofs);
  247. (* The format of the section information is:
  248. null terminated section name
  249. zeroes to adjust to 4 byte boundary
  250. 4 byte section data file pointer
  251. 4 byte section size *)
  252. Repeat
  253. Name := Get0String(e.f);
  254. alignAmount := 4 - ((length (Name) + 1) MOD 4);
  255. Skip (e.f,AlignAmount);
  256. if (Name = asecname) then
  257. begin
  258. secOfs := getInt32(e.f);
  259. secLen := getInt32(e.f);
  260. end else
  261. Skip(e.f,8);
  262. until (Name = '') or (Name = asecname);
  263. FindSectionNetwareNLM := (Name=asecname);
  264. end;
  265. {$endif}
  266. {****************************************************************************
  267. COFF
  268. ****************************************************************************}
  269. {$if defined(PE32) or defined(PE32PLUS) or defined(GO32V2)}
  270. type
  271. tcoffsechdr=packed record
  272. name : array[0..7] of char;
  273. vsize : longint;
  274. rvaofs : longint;
  275. datalen : longint;
  276. datapos : longint;
  277. relocpos : longint;
  278. lineno1 : longint;
  279. nrelocs : word;
  280. lineno2 : word;
  281. flags : longint;
  282. end;
  283. coffsymbol=packed record
  284. name : array[0..3] of char; { real is [0..7], which overlaps the strofs ! }
  285. strofs : longint;
  286. value : longint;
  287. section : smallint;
  288. empty : word;
  289. typ : byte;
  290. aux : byte;
  291. end;
  292. function FindSectionCoff(var e:TExeFile;const asecname:string;var secofs,seclen:longint):boolean;
  293. var
  294. i : longint;
  295. sechdr : tcoffsechdr;
  296. secname : string;
  297. secnamebuf : array[0..255] of char;
  298. code,
  299. oldofs,
  300. bufsize : longint;
  301. strofs : cardinal;
  302. begin
  303. FindSectionCoff:=false;
  304. { read section info }
  305. seek(e.f,e.sechdrofs);
  306. for i:=1 to e.nsects do
  307. begin
  308. blockread(e.f,sechdr,sizeof(sechdr),bufsize);
  309. move(sechdr.name,secnamebuf,8);
  310. secnamebuf[8]:=#0;
  311. secname:=strpas(secnamebuf);
  312. if secname[1]='/' then
  313. begin
  314. Val(Copy(secname,2,8),strofs,code);
  315. if code=0 then
  316. begin
  317. fillchar(secnamebuf,sizeof(secnamebuf),0);
  318. oldofs:=filepos(e.f);
  319. seek(e.f,e.secstrofs+strofs);
  320. blockread(e.f,secnamebuf,sizeof(secnamebuf),bufsize);
  321. seek(e.f,oldofs);
  322. secname:=strpas(secnamebuf);
  323. end
  324. else
  325. secname:='';
  326. end;
  327. if asecname=secname then
  328. begin
  329. secofs:=cardinal(sechdr.datapos) + E.ImgOffset;
  330. seclen:=sechdr.datalen;
  331. FindSectionCoff:=true;
  332. exit;
  333. end;
  334. end;
  335. end;
  336. {$endif PE32 or PE32PLUS or GO32V2}
  337. {$ifdef go32v2}
  338. function OpenGo32Coff(var e:TExeFile):boolean;
  339. type
  340. tgo32coffheader=packed record
  341. mach : word;
  342. nsects : word;
  343. time : longint;
  344. sympos : longint;
  345. syms : longint;
  346. opthdr : word;
  347. flag : word;
  348. other : array[0..27] of byte;
  349. end;
  350. const
  351. ParagraphSize = 512;
  352. var
  353. coffheader : tgo32coffheader;
  354. DosHeader: TDosHeader;
  355. BRead: cardinal;
  356. begin
  357. OpenGo32Coff:=false;
  358. { read and check header }
  359. if E.Size < SizeOf (DosHeader) then
  360. Exit;
  361. BlockRead (E.F, DosHeader, SizeOf (DosHeader), BRead);
  362. if BRead <> SizeOf (DosHeader) then
  363. Exit;
  364. if DosHeader.E_Magic = $5A4D then
  365. begin
  366. E.ImgOffset := DosHeader.e_cp * ParagraphSize;
  367. if DosHeader.e_cblp > 0 then
  368. E.ImgOffset := E.ImgOffset + DosHeader.e_cblp - ParagraphSize;
  369. end;
  370. if e.size < E.ImgOffset + sizeof(coffheader) then
  371. exit;
  372. seek(e.f,E.ImgOffset);
  373. blockread(e.f,coffheader,sizeof(coffheader));
  374. if coffheader.mach<>$14c then
  375. exit;
  376. e.sechdrofs:=filepos(e.f);
  377. e.nsects:=coffheader.nsects;
  378. e.secstrofs:=coffheader.sympos+coffheader.syms*sizeof(coffsymbol)+4;
  379. if e.secstrofs>e.size then
  380. exit;
  381. OpenGo32Coff:=true;
  382. end;
  383. {$endif Go32v2}
  384. {$ifdef PE32}
  385. function OpenPeCoff(var e:TExeFile):boolean;
  386. type
  387. tpeheader = packed record
  388. PEMagic : longint;
  389. Machine : word;
  390. NumberOfSections : word;
  391. TimeDateStamp : longint;
  392. PointerToSymbolTable : longint;
  393. NumberOfSymbols : longint;
  394. SizeOfOptionalHeader : word;
  395. Characteristics : word;
  396. Magic : word;
  397. MajorLinkerVersion : byte;
  398. MinorLinkerVersion : byte;
  399. SizeOfCode : longint;
  400. SizeOfInitializedData : longint;
  401. SizeOfUninitializedData : longint;
  402. AddressOfEntryPoint : longint;
  403. BaseOfCode : longint;
  404. BaseOfData : longint;
  405. ImageBase : longint;
  406. SectionAlignment : longint;
  407. FileAlignment : longint;
  408. MajorOperatingSystemVersion : word;
  409. MinorOperatingSystemVersion : word;
  410. MajorImageVersion : word;
  411. MinorImageVersion : word;
  412. MajorSubsystemVersion : word;
  413. MinorSubsystemVersion : word;
  414. Reserved1 : longint;
  415. SizeOfImage : longint;
  416. SizeOfHeaders : longint;
  417. CheckSum : longint;
  418. Subsystem : word;
  419. DllCharacteristics : word;
  420. SizeOfStackReserve : longint;
  421. SizeOfStackCommit : longint;
  422. SizeOfHeapReserve : longint;
  423. SizeOfHeapCommit : longint;
  424. LoaderFlags : longint;
  425. NumberOfRvaAndSizes : longint;
  426. DataDirectory : array[1..$80] of byte;
  427. end;
  428. var
  429. dosheader : tdosheader;
  430. peheader : tpeheader;
  431. begin
  432. OpenPeCoff:=false;
  433. { read and check header }
  434. if e.size<sizeof(dosheader) then
  435. exit;
  436. blockread(e.f,dosheader,sizeof(tdosheader));
  437. seek(e.f,dosheader.e_lfanew);
  438. blockread(e.f,peheader,sizeof(tpeheader));
  439. if peheader.pemagic<>$4550 then
  440. exit;
  441. e.sechdrofs:=filepos(e.f);
  442. e.nsects:=peheader.NumberOfSections;
  443. e.secstrofs:=peheader.PointerToSymbolTable+peheader.NumberOfSymbols*sizeof(coffsymbol);
  444. if e.secstrofs>e.size then
  445. exit;
  446. e.processaddress:=peheader.ImageBase;
  447. OpenPeCoff:=true;
  448. end;
  449. {$endif PE32}
  450. {$ifdef PE32PLUS}
  451. function OpenPePlusCoff(var e:TExeFile):boolean;
  452. type
  453. tpeheader = packed record
  454. PEMagic : longint;
  455. Machine : word;
  456. NumberOfSections : word;
  457. TimeDateStamp : longint;
  458. PointerToSymbolTable : longint;
  459. NumberOfSymbols : longint;
  460. SizeOfOptionalHeader : word;
  461. Characteristics : word;
  462. Magic : word;
  463. MajorLinkerVersion : byte;
  464. MinorLinkerVersion : byte;
  465. SizeOfCode : longint;
  466. SizeOfInitializedData : longint;
  467. SizeOfUninitializedData : longint;
  468. AddressOfEntryPoint : longint;
  469. BaseOfCode : longint;
  470. ImageBase : qword;
  471. SectionAlignment : longint;
  472. FileAlignment : longint;
  473. MajorOperatingSystemVersion : word;
  474. MinorOperatingSystemVersion : word;
  475. MajorImageVersion : word;
  476. MinorImageVersion : word;
  477. MajorSubsystemVersion : word;
  478. MinorSubsystemVersion : word;
  479. Reserved1 : longint;
  480. SizeOfImage : longint;
  481. SizeOfHeaders : longint;
  482. CheckSum : longint;
  483. Subsystem : word;
  484. DllCharacteristics : word;
  485. SizeOfStackReserve : qword;
  486. SizeOfStackCommit : qword;
  487. SizeOfHeapReserve : qword;
  488. SizeOfHeapCommit : qword;
  489. LoaderFlags : longint;
  490. NumberOfRvaAndSizes : longint;
  491. DataDirectory : array[1..$80] of byte;
  492. end;
  493. var
  494. dosheader : tdosheader;
  495. peheader : tpeheader;
  496. begin
  497. OpenPePlusCoff:=false;
  498. { read and check header }
  499. if E.Size<sizeof(dosheader) then
  500. exit;
  501. blockread(E.F,dosheader,sizeof(tdosheader));
  502. seek(E.F,dosheader.e_lfanew);
  503. blockread(E.F,peheader,sizeof(tpeheader));
  504. if peheader.pemagic<>$4550 then
  505. exit;
  506. e.sechdrofs:=filepos(e.f);
  507. e.nsects:=peheader.NumberOfSections;
  508. e.secstrofs:=peheader.PointerToSymbolTable+peheader.NumberOfSymbols*sizeof(coffsymbol);
  509. if e.secstrofs>e.size then
  510. exit;
  511. e.processaddress:=peheader.ImageBase;
  512. OpenPePlusCoff:=true;
  513. end;
  514. {$endif PE32PLUS}
  515. {****************************************************************************
  516. AOUT
  517. ****************************************************************************}
  518. {$IFDEF EMX}
  519. type
  520. TEmxHeader = packed record
  521. Version: array [1..16] of char;
  522. Bound: word;
  523. AoutOfs: longint;
  524. Options: array [1..42] of char;
  525. end;
  526. TAoutHeader = packed record
  527. Magic: word;
  528. Machine: byte;
  529. Flags: byte;
  530. TextSize: longint;
  531. DataSize: longint;
  532. BssSize: longint;
  533. SymbSize: longint;
  534. EntryPoint: longint;
  535. TextRelocSize: longint;
  536. DataRelocSize: longint;
  537. end;
  538. const
  539. StartPageSize = $1000;
  540. var
  541. DosHeader: TDosHeader;
  542. EmxHeader: TEmxHeader;
  543. AoutHeader: TAoutHeader;
  544. StabOfs: PtrUInt;
  545. S4: string [4];
  546. function OpenEMXaout (var E: TExeFile): boolean;
  547. begin
  548. OpenEMXaout := false;
  549. { GDB after 4.18 uses offset to function begin
  550. in text section but OS/2 version still uses 4.16 PM }
  551. E.FunctionRelative := false;
  552. { read and check header }
  553. if E.Size > SizeOf (DosHeader) then
  554. begin
  555. BlockRead (E.F, DosHeader, SizeOf (TDosHeader));
  556. if E.Size > DosHeader.e_cparhdr shl 4 + SizeOf (TEmxHeader) then
  557. begin
  558. Seek (E.F, DosHeader.e_cparhdr shl 4);
  559. BlockRead (E.F, EmxHeader, SizeOf (TEmxHeader));
  560. S4 [0] := #4;
  561. Move (EmxHeader.Version, S4 [1], 4);
  562. if (S4 = 'emx ') and
  563. (E.Size > EmxHeader.AoutOfs + SizeOf (TAoutHeader)) then
  564. begin
  565. Seek (E.F, EmxHeader.AoutOfs);
  566. BlockRead (E.F, AoutHeader, SizeOf (TAoutHeader));
  567. if AOutHeader.Magic = $10B then
  568. StabOfs := StartPageSize
  569. else
  570. StabOfs := EmxHeader.AoutOfs + SizeOf (TAoutHeader);
  571. StabOfs := StabOfs
  572. + AoutHeader.TextSize
  573. + AoutHeader.DataSize
  574. + AoutHeader.TextRelocSize
  575. + AoutHeader.DataRelocSize;
  576. if E.Size > StabOfs + AoutHeader.SymbSize then
  577. OpenEMXaout := true;
  578. end;
  579. end;
  580. end;
  581. end;
  582. function FindSectionEMXaout (var E: TExeFile; const ASecName: string;
  583. var SecOfs, SecLen: longint): boolean;
  584. begin
  585. FindSectionEMXaout := false;
  586. if ASecName = '.stab' then
  587. begin
  588. SecOfs := StabOfs;
  589. SecLen := AoutHeader.SymbSize;
  590. FindSectionEMXaout := true;
  591. end else
  592. if ASecName = '.stabstr' then
  593. begin
  594. SecOfs := StabOfs + AoutHeader.SymbSize;
  595. SecLen := E.Size - Pred (SecOfs);
  596. FindSectionEMXaout := true;
  597. end;
  598. end;
  599. {$ENDIF EMX}
  600. {****************************************************************************
  601. ELF
  602. ****************************************************************************}
  603. {$if defined(ELF32) or defined(BEOS)}
  604. type
  605. telfheader=packed record
  606. magic0123 : longint;
  607. file_class : byte;
  608. data_encoding : byte;
  609. file_version : byte;
  610. padding : array[$07..$0f] of byte;
  611. e_type : word;
  612. e_machine : word;
  613. e_version : longword;
  614. e_entry : longword; // entrypoint
  615. e_phoff : longword; // program header offset
  616. e_shoff : longword; // sections header offset
  617. e_flags : longword;
  618. e_ehsize : word; // elf header size in bytes
  619. e_phentsize : word; // size of an entry in the program header array
  620. e_phnum : word; // 0..e_phnum-1 of entrys
  621. e_shentsize : word; // size of an entry in sections header array
  622. e_shnum : word; // 0..e_shnum-1 of entrys
  623. e_shstrndx : word; // index of string section header
  624. end;
  625. telfsechdr=packed record
  626. sh_name : longword;
  627. sh_type : longword;
  628. sh_flags : longword;
  629. sh_addr : longword;
  630. sh_offset : longword;
  631. sh_size : longword;
  632. sh_link : longword;
  633. sh_info : longword;
  634. sh_addralign : longword;
  635. sh_entsize : longword;
  636. end;
  637. {$endif ELF32 or BEOS}
  638. {$ifdef ELF64}
  639. type
  640. telfheader=packed record
  641. magic0123 : longint;
  642. file_class : byte;
  643. data_encoding : byte;
  644. file_version : byte;
  645. padding : array[$07..$0f] of byte;
  646. e_type : word;
  647. e_machine : word;
  648. e_version : longword;
  649. e_entry : int64; // entrypoint
  650. e_phoff : int64; // program header offset
  651. e_shoff : int64; // sections header offset
  652. e_flags : longword;
  653. e_ehsize : word; // elf header size in bytes
  654. e_phentsize : word; // size of an entry in the program header array
  655. e_phnum : word; // 0..e_phnum-1 of entrys
  656. e_shentsize : word; // size of an entry in sections header array
  657. e_shnum : word; // 0..e_shnum-1 of entrys
  658. e_shstrndx : word; // index of string section header
  659. end;
  660. type
  661. telfsechdr=packed record
  662. sh_name : longword;
  663. sh_type : longword;
  664. sh_flags : int64;
  665. sh_addr : int64;
  666. sh_offset : int64;
  667. sh_size : int64;
  668. sh_link : longword;
  669. sh_info : longword;
  670. sh_addralign : int64;
  671. sh_entsize : int64;
  672. end;
  673. {$endif ELF64}
  674. {$if defined(ELF32) or defined(ELF64) or defined(BEOS)}
  675. function OpenElf(var e:TExeFile):boolean;
  676. var
  677. elfheader : telfheader;
  678. elfsec : telfsechdr;
  679. begin
  680. OpenElf:=false;
  681. { read and check header }
  682. if e.size<sizeof(telfheader) then
  683. exit;
  684. blockread(e.f,elfheader,sizeof(telfheader));
  685. if elfheader.magic0123<>{$ifdef ENDIAN_LITTLE}$464c457f{$else}$7f454c46{$endif} then
  686. exit;
  687. if elfheader.e_shentsize<>sizeof(telfsechdr) then
  688. exit;
  689. { read section names }
  690. seek(e.f,elfheader.e_shoff+elfheader.e_shstrndx*cardinal(sizeof(telfsechdr)));
  691. blockread(e.f,elfsec,sizeof(telfsechdr));
  692. e.secstrofs:=elfsec.sh_offset;
  693. e.sechdrofs:=elfheader.e_shoff;
  694. e.nsects:=elfheader.e_shnum;
  695. OpenElf:=true;
  696. end;
  697. function FindSectionElf(var e:TExeFile;const asecname:string;var secofs,seclen:longint):boolean;
  698. var
  699. elfsec : telfsechdr;
  700. secname : string;
  701. secnamebuf : array[0..255] of char;
  702. oldofs,
  703. bufsize,i : longint;
  704. begin
  705. FindSectionElf:=false;
  706. seek(e.f,e.sechdrofs);
  707. for i:=1 to e.nsects do
  708. begin
  709. blockread(e.f,elfsec,sizeof(telfsechdr));
  710. fillchar(secnamebuf,sizeof(secnamebuf),0);
  711. oldofs:=filepos(e.f);
  712. seek(e.f,e.secstrofs+elfsec.sh_name);
  713. blockread(e.f,secnamebuf,sizeof(secnamebuf)-1,bufsize);
  714. seek(e.f,oldofs);
  715. secname:=strpas(secnamebuf);
  716. if asecname=secname then
  717. begin
  718. secofs:=elfsec.sh_offset;
  719. seclen:=elfsec.sh_size;
  720. FindSectionElf:=true;
  721. exit;
  722. end;
  723. end;
  724. end;
  725. {$endif ELF32 or ELF64 or BEOS}
  726. {$ifdef beos}
  727. {$i ptypes.inc}
  728. type
  729. // Descriptive formats
  730. status_t = Longint;
  731. team_id = Longint;
  732. image_id = Longint;
  733. { image types }
  734. const
  735. B_APP_IMAGE = 1;
  736. B_LIBRARY_IMAGE = 2;
  737. B_ADD_ON_IMAGE = 3;
  738. B_SYSTEM_IMAGE = 4;
  739. B_OK = 0;
  740. type
  741. image_info = packed record
  742. id : image_id;
  743. _type : longint;
  744. sequence: longint;
  745. init_order: longint;
  746. init_routine: pointer;
  747. term_routine: pointer;
  748. device: dev_t;
  749. node: ino_t;
  750. name: array[0..MAXPATHLEN-1] of char;
  751. { name: string[255];
  752. name2: string[255];
  753. name3: string[255];
  754. name4: string[255];
  755. name5: string[5];
  756. }
  757. text: pointer;
  758. data: pointer;
  759. text_size: longint;
  760. data_size: longint;
  761. end;
  762. function get_next_image_info(team: team_id; var cookie:longint; var info:image_info; size: size_t) : status_t;cdecl; external 'root' name '_get_next_image_info';
  763. function OpenElf32Beos(var e:TExeFile):boolean;
  764. var
  765. cookie : longint;
  766. info : image_info;
  767. begin
  768. // The only BeOS specific part is setting the processaddress
  769. cookie := 0;
  770. OpenElf32Beos:=false;
  771. fillchar(info, sizeof(image_info), 0);
  772. while get_next_image_info(0,cookie,info,sizeof(info))=B_OK do
  773. begin
  774. if e.filename=String(pchar(@info.name)) then
  775. begin
  776. if (info._type = B_APP_IMAGE) then
  777. e.processaddress := cardinal(info.text)
  778. else
  779. e.processaddress := 0;
  780. OpenElf32Beos := OpenElf(e);
  781. exit;
  782. end;
  783. end;
  784. end;
  785. {$endif beos}
  786. {****************************************************************************
  787. MACHO
  788. ****************************************************************************}
  789. {$ifdef darwin}
  790. type
  791. MachoFatHeader= packed record
  792. magic: longint;
  793. nfatarch: longint;
  794. end;
  795. MachoHeader=packed record
  796. magic: longword;
  797. cpu_type_t: longint;
  798. cpu_subtype_t: longint;
  799. filetype: longint;
  800. ncmds: longint;
  801. sizeofcmds: longint;
  802. flags: longint;
  803. end;
  804. cmdblock=packed record
  805. cmd: longint;
  806. cmdsize: longint;
  807. end;
  808. symbSeg=packed record
  809. symoff : longint;
  810. nsyms : longint;
  811. stroff : longint;
  812. strsize: longint;
  813. end;
  814. tstab=packed record
  815. strpos : longint;
  816. ntype : byte;
  817. nother : byte;
  818. ndesc : word;
  819. nvalue : dword;
  820. end;
  821. function OpenMachO32PPC(var e:TExeFile):boolean;
  822. var
  823. mh:MachoHeader;
  824. begin
  825. OpenMachO32PPC:= false;
  826. E.FunctionRelative:=false;
  827. if e.size<sizeof(mh) then
  828. exit;
  829. blockread (e.f, mh, sizeof(mh));
  830. e.sechdrofs:=filepos(e.f);
  831. e.nsects:=mh.ncmds;
  832. OpenMachO32PPC:=true;
  833. end;
  834. function FindSectionMachO32PPC(var e:TExeFile;const asecname:string;var secofs,seclen:longint):boolean;
  835. var
  836. i: longint;
  837. block:cmdblock;
  838. symbolsSeg: symbSeg;
  839. begin
  840. FindSectionMachO32PPC:=false;
  841. seek(e.f,e.sechdrofs);
  842. for i:= 1 to e.nsects do
  843. begin
  844. {$I-}
  845. blockread (e.f, block, sizeof(block));
  846. {$I+}
  847. if IOResult <> 0 then
  848. Exit;
  849. if block.cmd = $2 then
  850. begin
  851. blockread (e.f, symbolsSeg, sizeof(symbolsSeg));
  852. if asecname='.stab' then
  853. begin
  854. secofs:=symbolsSeg.symoff;
  855. { the caller will divide again by sizeof(tstab) }
  856. seclen:=symbolsSeg.nsyms*sizeof(tstab);
  857. FindSectionMachO32PPC:=true;
  858. end
  859. else if asecname='.stabstr' then
  860. begin
  861. secofs:=symbolsSeg.stroff;
  862. seclen:=symbolsSeg.strsize;
  863. FindSectionMachO32PPC:=true;
  864. end;
  865. exit;
  866. end;
  867. Seek(e.f, FilePos (e.f) + block.cmdsize - sizeof(block));
  868. end;
  869. end;
  870. {$endif darwin}
  871. {****************************************************************************
  872. CRC
  873. ****************************************************************************}
  874. var
  875. Crc32Tbl : array[0..255] of cardinal;
  876. procedure MakeCRC32Tbl;
  877. var
  878. crc : cardinal;
  879. i,n : integer;
  880. begin
  881. for i:=0 to 255 do
  882. begin
  883. crc:=i;
  884. for n:=1 to 8 do
  885. if (crc and 1)<>0 then
  886. crc:=(crc shr 1) xor cardinal($edb88320)
  887. else
  888. crc:=crc shr 1;
  889. Crc32Tbl[i]:=crc;
  890. end;
  891. end;
  892. Function UpdateCrc32(InitCrc:cardinal;const InBuf;InLen:LongInt):cardinal;
  893. var
  894. i : LongInt;
  895. p : pchar;
  896. begin
  897. if Crc32Tbl[1]=0 then
  898. MakeCrc32Tbl;
  899. p:=@InBuf;
  900. UpdateCrc32:=not InitCrc;
  901. for i:=1 to InLen do
  902. begin
  903. UpdateCrc32:=Crc32Tbl[byte(UpdateCrc32) xor byte(p^)] xor (UpdateCrc32 shr 8);
  904. inc(p);
  905. end;
  906. UpdateCrc32:=not UpdateCrc32;
  907. end;
  908. {****************************************************************************
  909. Generic Executable Open/Close
  910. ****************************************************************************}
  911. type
  912. TOpenProc=function(var e:TExeFile):boolean;
  913. TFindSectionProc=function(var e:TExeFile;const asecname:string;var secofs,seclen:longint):boolean;
  914. TExeProcRec=record
  915. openproc : TOpenProc;
  916. findproc : TFindSectionProc;
  917. end;
  918. const
  919. ExeProcs : TExeProcRec = (
  920. {$ifdef go32v2}
  921. openproc : @OpenGo32Coff;
  922. findproc : @FindSectionCoff;
  923. {$endif}
  924. {$ifdef PE32}
  925. openproc : @OpenPeCoff;
  926. findproc : @FindSectionCoff;
  927. {$endif}
  928. {$ifdef PE32PLUS}
  929. openproc : @OpenPePlusCoff;
  930. findproc : @FindSectionCoff;
  931. {$endif PE32PLUS}
  932. {$if defined(ELF32) or defined(ELF64)}
  933. openproc : @OpenElf;
  934. findproc : @FindSectionElf;
  935. {$endif ELF32 or ELF64}
  936. {$ifdef BEOS}
  937. openproc : @OpenElf32Beos;
  938. findproc : @FindSectionElf;
  939. {$endif BEOS}
  940. {$ifdef darwin}
  941. openproc : @OpenMachO32PPC;
  942. findproc : @FindSectionMachO32PPC;
  943. {$endif darwin}
  944. {$IFDEF EMX}
  945. openproc : @OpenEMXaout;
  946. findproc : @FindSectionEMXaout;
  947. {$ENDIF EMX}
  948. {$ifdef netware}
  949. openproc : @OpenNetwareNLM;
  950. findproc : @FindSectionNetwareNLM;
  951. {$endif}
  952. );
  953. function OpenExeFile(var e:TExeFile;const fn:string):boolean;
  954. var
  955. ofm : word;
  956. begin
  957. OpenExeFile:=false;
  958. fillchar(e,sizeof(e),0);
  959. e.bufsize:=sizeof(e.buf);
  960. e.filename:=fn;
  961. if fn='' then // we don't want to read stdin
  962. exit;
  963. assign(e.f,fn);
  964. {$I-}
  965. ofm:=filemode;
  966. filemode:=$40;
  967. reset(e.f,1);
  968. filemode:=ofm;
  969. {$I+}
  970. if ioresult<>0 then
  971. exit;
  972. e.isopen:=true;
  973. // cache filesize
  974. e.size:=filesize(e.f);
  975. E.FunctionRelative := true;
  976. E.ImgOffset := 0;
  977. if ExeProcs.OpenProc<>nil then
  978. OpenExeFile:=ExeProcs.OpenProc(e);
  979. end;
  980. function CloseExeFile(var e:TExeFile):boolean;
  981. begin
  982. CloseExeFile:=false;
  983. if not e.isopen then
  984. exit;
  985. e.isopen:=false;
  986. close(e.f);
  987. CloseExeFile:=true;
  988. end;
  989. function FindExeSection(var e:TExeFile;const secname:string;var secofs,seclen:longint):boolean;
  990. begin
  991. FindExeSection:=false;
  992. if not e.isopen then
  993. exit;
  994. if ExeProcs.FindProc<>nil then
  995. FindExeSection:=ExeProcs.FindProc(e,secname,secofs,seclen);
  996. end;
  997. function CheckDbgFile(var e:TExeFile;const fn:string;dbgcrc:cardinal):boolean;
  998. var
  999. c : cardinal;
  1000. ofm : word;
  1001. g : file;
  1002. begin
  1003. CheckDbgFile:=false;
  1004. assign(g,fn);
  1005. {$I-}
  1006. ofm:=filemode;
  1007. filemode:=$40;
  1008. reset(g,1);
  1009. filemode:=ofm;
  1010. {$I+}
  1011. if ioresult<>0 then
  1012. exit;
  1013. { We reuse the buffer from e here to prevent too much stack allocation }
  1014. c:=0;
  1015. repeat
  1016. blockread(g,e.buf,e.bufsize,e.bufcnt);
  1017. c:=UpdateCrc32(c,e.buf,e.bufcnt);
  1018. until e.bufcnt<e.bufsize;
  1019. close(g);
  1020. CheckDbgFile:=(dbgcrc=c);
  1021. end;
  1022. function ReadDebugLink(var e:TExeFile;var dbgfn:string):boolean;
  1023. var
  1024. dbglink : array[0..255] of char;
  1025. i,
  1026. dbglinklen,
  1027. dbglinkofs : longint;
  1028. dbgcrc : cardinal;
  1029. begin
  1030. ReadDebugLink:=false;
  1031. if not FindExeSection(e,'.gnu_debuglink',dbglinkofs,dbglinklen) then
  1032. exit;
  1033. if dbglinklen>sizeof(dbglink)-1 then
  1034. exit;
  1035. fillchar(dbglink,sizeof(dbglink),0);
  1036. seek(e.f,dbglinkofs);
  1037. blockread(e.f,dbglink,dbglinklen);
  1038. dbgfn:=strpas(dbglink);
  1039. if length(dbgfn)=0 then
  1040. exit;
  1041. i:=align(length(dbgfn)+1,4);
  1042. if (i+4)>dbglinklen then
  1043. exit;
  1044. move(dbglink[i],dbgcrc,4);
  1045. { current dir }
  1046. if CheckDbgFile(e,dbgfn,dbgcrc) then
  1047. begin
  1048. ReadDebugLink:=true;
  1049. exit;
  1050. end;
  1051. { executable dir }
  1052. i:=length(e.filename);
  1053. while (i>0) and not(e.filename[i] in AllowDirectorySeparators) do
  1054. dec(i);
  1055. if i>0 then
  1056. begin
  1057. dbgfn:=copy(e.filename,1,i)+dbgfn;
  1058. if CheckDbgFile(e,dbgfn,dbgcrc) then
  1059. begin
  1060. ReadDebugLink:=true;
  1061. exit;
  1062. end;
  1063. end;
  1064. end;
  1065. end.