2
0

owar.pas 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548
  1. {
  2. Copyright (c) 1998-2002 by Peter Vreman
  3. Contains the stuff for writing .a files directly
  4. This program is free software; you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation; either version 2 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program; if not, write to the Free Software
  14. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  15. ****************************************************************************
  16. }
  17. unit owar;
  18. {$i fpcdefs.inc}
  19. interface
  20. uses
  21. cclasses,
  22. owbase;
  23. type
  24. tarhdr=packed record
  25. name : array[0..15] of char;
  26. date : array[0..11] of char;
  27. uid : array[0..5] of char;
  28. gid : array[0..5] of char;
  29. mode : array[0..7] of char;
  30. size : array[0..9] of char;
  31. fmag : array[0..1] of char;
  32. end;
  33. tarobjectwriter=class(tobjectwriter)
  34. constructor createAr(const Aarfn:string);override;
  35. destructor destroy;override;
  36. function createfile(const fn:string):boolean;override;
  37. procedure closefile;override;
  38. procedure writesym(const sym:string);override;
  39. procedure write(const b;len:longword);override;
  40. private
  41. arfn : string;
  42. arhdr : tarhdr;
  43. symreloc,
  44. symstr,
  45. lfnstr,
  46. ardata : TDynamicArray;
  47. objpos : longint;
  48. objfn : string;
  49. timestamp : string[12];
  50. procedure createarhdr(fn:string;asize:longint;const gid,uid,mode:string);
  51. procedure writear;
  52. end;
  53. tarobjectreader=class(tobjectreader)
  54. private
  55. ArSymbols : TFPHashObjectList;
  56. LFNStrs : PChar;
  57. LFNSize : longint;
  58. CurrMemberPos,
  59. CurrMemberSize : longint;
  60. CurrMemberName : string;
  61. isar: boolean;
  62. function DecodeMemberName(ahdr:TArHdr):string;
  63. function DecodeMemberSize(ahdr:TArHdr):longint;
  64. procedure ReadArchive;
  65. protected
  66. function getfilename:string;override;
  67. function GetPos: longint;override;
  68. function GetIsArchive: boolean; override;
  69. public
  70. constructor createAr(const Aarfn:string;allow_nonar:boolean=false);override;
  71. destructor destroy;override;
  72. function openfile(const fn:string):boolean;override;
  73. procedure closefile;override;
  74. procedure seek(len:longint);override;
  75. end;
  76. implementation
  77. uses
  78. SysUtils,
  79. cstreams,
  80. systems,
  81. globals,
  82. verbose;
  83. const
  84. symrelocbufsize = 4096;
  85. symstrbufsize = 8192;
  86. lfnstrbufsize = 4096;
  87. arbufsize = 65536;
  88. armagic:array[1..8] of char='!<arch>'#10;
  89. type
  90. TArSymbol = class(TFPHashObject)
  91. MemberPos : longint;
  92. end;
  93. {*****************************************************************************
  94. Helpers
  95. *****************************************************************************}
  96. const
  97. C1970=2440588;
  98. D0=1461;
  99. D1=146097;
  100. D2=1721119;
  101. Function Gregorian2Julian(DT:TSystemTime):LongInt;
  102. Var
  103. Century,XYear,Month : LongInt;
  104. Begin
  105. Month:=DT.Month;
  106. If Month<=2 Then
  107. Begin
  108. Dec(DT.Year);
  109. Inc(Month,12);
  110. End;
  111. Dec(Month,3);
  112. Century:=(longint(DT.Year Div 100)*D1) shr 2;
  113. XYear:=(longint(DT.Year Mod 100)*D0) shr 2;
  114. Gregorian2Julian:=((((Month*153)+2) div 5)+DT.Day)+D2+XYear+Century;
  115. End;
  116. function DT2Unix(DT:TSystemTime):LongInt;
  117. Begin
  118. DT2Unix:=(Gregorian2Julian(DT)-C1970)*86400+(LongInt(DT.Hour)*3600)+(DT.Minute*60)+DT.Second;
  119. end;
  120. function lsb2msb(l:longint):longint;
  121. type
  122. bytearr=array[0..3] of byte;
  123. begin
  124. {$ifndef FPC_BIG_ENDIAN}
  125. bytearr(result)[0]:=bytearr(l)[3];
  126. bytearr(result)[1]:=bytearr(l)[2];
  127. bytearr(result)[2]:=bytearr(l)[1];
  128. bytearr(result)[3]:=bytearr(l)[0];
  129. {$else}
  130. result:=l;
  131. {$endif}
  132. end;
  133. {*****************************************************************************
  134. TArObjectWriter
  135. *****************************************************************************}
  136. constructor tarobjectwriter.createAr(const Aarfn:string);
  137. var
  138. time : TSystemTime;
  139. begin
  140. arfn:=Aarfn;
  141. ardata:=TDynamicArray.Create(arbufsize);
  142. symreloc:=TDynamicArray.Create(symrelocbufsize);
  143. symstr:=TDynamicArray.Create(symstrbufsize);
  144. lfnstr:=TDynamicArray.Create(lfnstrbufsize);
  145. { create timestamp }
  146. GetLocalTime(time);
  147. Str(DT2Unix(time),timestamp);
  148. end;
  149. destructor tarobjectwriter.destroy;
  150. begin
  151. if Errorcount=0 then
  152. writear;
  153. arData.Free;
  154. symreloc.Free;
  155. symstr.Free;
  156. lfnstr.Free;
  157. end;
  158. procedure tarobjectwriter.createarhdr(fn:string;asize:longint;const gid,uid,mode:string);
  159. var
  160. tmp : string[9];
  161. hfn : string;
  162. begin
  163. { create ar header }
  164. fillchar(arhdr,sizeof(tarhdr),' ');
  165. { win32 will change names starting with .\ to ./ when using lfn, corrupting
  166. the sort order required for the idata sections. To prevent this strip
  167. always the path from the filename. (PFV) }
  168. hfn:=ExtractFileName(fn);
  169. if hfn='' then
  170. hfn:=fn;
  171. fn:=hfn+'/';
  172. if length(fn)>16 then
  173. begin
  174. arhdr.name[0]:='/';
  175. str(lfnstr.size,tmp);
  176. move(tmp[1],arhdr.name[1],length(tmp));
  177. fn:=fn+#10;
  178. lfnstr.write(fn[1],length(fn));
  179. end
  180. else
  181. move(fn[1],arhdr.name,length(fn));
  182. { don't write a date if also no gid/uid/mode is specified }
  183. if gid<>'' then
  184. move(timestamp[1],arhdr.date,length(timestamp));
  185. str(asize,tmp);
  186. move(tmp[1],arhdr.size,length(tmp));
  187. move(gid[1],arhdr.gid,length(gid));
  188. move(uid[1],arhdr.uid,length(uid));
  189. move(mode[1],arhdr.mode,length(mode));
  190. arhdr.fmag:='`'#10;
  191. end;
  192. function tarobjectwriter.createfile(const fn:string):boolean;
  193. begin
  194. objfn:=fn;
  195. objpos:=ardata.size;
  196. ardata.seek(objpos + sizeof(tarhdr));
  197. createfile:=true;
  198. fobjsize:=0;
  199. end;
  200. procedure tarobjectwriter.closefile;
  201. const
  202. LF:char=#10;
  203. var
  204. filesize:longint;
  205. begin
  206. { preserve file size, before aligning on an even boundary }
  207. filesize:=ardata.size-objpos-sizeof(tarhdr);
  208. { align on an even boundary, by inserting an LF if necessary }
  209. if odd(ardata.size) then
  210. write(LF,1);
  211. { fix the size in the header }
  212. createarhdr(objfn,filesize,'42','42','644');
  213. { write the header }
  214. ardata.seek(objpos);
  215. ardata.write(arhdr,sizeof(tarhdr));
  216. fobjsize:=0;
  217. end;
  218. procedure tarobjectwriter.writesym(const sym:string);
  219. var
  220. c : char;
  221. begin
  222. c:=#0;
  223. symreloc.write(objpos,4);
  224. symstr.write(sym[1],length(sym));
  225. symstr.write(c,1);
  226. end;
  227. procedure tarobjectwriter.write(const b;len:longword);
  228. begin
  229. inc(fobjsize,len);
  230. inc(fsize,len);
  231. ardata.write(b,len);
  232. end;
  233. procedure tarobjectwriter.writear;
  234. var
  235. arf : TCCustomFileStream;
  236. fixup,l,
  237. relocs,i : longint;
  238. begin
  239. arf:=CFileStreamClass.Create(arfn,fmCreate);
  240. if CStreamError<>0 then
  241. begin
  242. Message1(exec_e_cant_create_archivefile,arfn);
  243. exit;
  244. end;
  245. arf.Write(armagic,sizeof(armagic));
  246. { align first, because we need the size for the fixups of the symbol reloc }
  247. if lfnstr.size>0 then
  248. lfnstr.align(2);
  249. if symreloc.size>0 then
  250. begin
  251. symstr.align(2);
  252. fixup:=12+sizeof(tarhdr)+symreloc.size+symstr.size;
  253. if lfnstr.size>0 then
  254. inc(fixup,lfnstr.size+sizeof(tarhdr));
  255. relocs:=symreloc.size div 4;
  256. { fixup relocs }
  257. for i:=0to relocs-1 do
  258. begin
  259. symreloc.seek(i*4);
  260. symreloc.read(l,4);
  261. symreloc.seek(i*4);
  262. l:=lsb2msb(l+fixup);
  263. symreloc.write(l,4);
  264. end;
  265. createarhdr('',4+symreloc.size+symstr.size,'0','0','0');
  266. arf.Write(arhdr,sizeof(tarhdr));
  267. relocs:=lsb2msb(relocs);
  268. arf.Write(relocs,4);
  269. symreloc.WriteStream(arf);
  270. symstr.WriteStream(arf);
  271. end;
  272. if lfnstr.size>0 then
  273. begin
  274. createarhdr('/',lfnstr.size,'','','');
  275. arf.Write(arhdr,sizeof(tarhdr));
  276. lfnstr.WriteStream(arf);
  277. end;
  278. ardata.WriteStream(arf);
  279. Arf.Free;
  280. end;
  281. {*****************************************************************************
  282. TArObjectReader
  283. *****************************************************************************}
  284. constructor tarobjectreader.createAr(const Aarfn:string;allow_nonar:boolean);
  285. var
  286. magic:array[0..sizeof(armagic)-1] of char;
  287. begin
  288. inherited Create;
  289. ArSymbols:=TFPHashObjectList.Create(true);
  290. CurrMemberPos:=0;
  291. CurrMemberSize:=0;
  292. CurrMemberName:='';
  293. if inherited openfile(Aarfn) then
  294. begin
  295. Read(magic,sizeof(armagic));
  296. isar:=(CompareByte(magic,armagic,sizeof(armagic))=0);
  297. if isar then
  298. ReadArchive
  299. else if (not allow_nonar) then
  300. Comment(V_Error,'Not a ar file, illegal magic: '+filename);
  301. Seek(0);
  302. end;
  303. end;
  304. destructor tarobjectreader.destroy;
  305. begin
  306. inherited closefile;
  307. ArSymbols.Free;
  308. if assigned(LFNStrs) then
  309. FreeMem(LFNStrs);
  310. inherited Destroy;
  311. end;
  312. function tarobjectreader.getfilename : string;
  313. begin
  314. result:=inherited getfilename;
  315. if CurrMemberName<>'' then
  316. result:=result+'('+CurrMemberName+')';
  317. end;
  318. function tarobjectreader.GetPos: longint;
  319. begin
  320. result:=inherited GetPos-CurrMemberPos;
  321. end;
  322. function tarobjectreader.GetIsArchive: boolean;
  323. begin
  324. Result:=isar;
  325. end;
  326. function tarobjectreader.DecodeMemberName(ahdr:TArHdr):string;
  327. var
  328. hs : string;
  329. code : integer;
  330. hsp,
  331. p : pchar;
  332. lfnidx : longint;
  333. begin
  334. result:='';
  335. p:[email protected][0];
  336. hsp:=@hs[1];
  337. while (p^<>' ') and (hsp-@hs[1]<16) do
  338. begin
  339. hsp^:=p^;
  340. inc(p);
  341. inc(hsp);
  342. end;
  343. hs[0]:=chr(hsp-@hs[1]);
  344. if (hs[1]='/') and (hs[2] in ['0'..'9']) then
  345. begin
  346. Delete(hs,1,1);
  347. val(hs,lfnidx,code);
  348. if (lfnidx<0) or (lfnidx>=LFNSize) then
  349. begin
  350. Comment(V_Error,'Invalid ar member lfn name index in '+filename);
  351. exit;
  352. end;
  353. p:=@LFNStrs[lfnidx];
  354. hsp:=@result[1];
  355. while p^<>#10 do
  356. begin
  357. hsp^:=p^;
  358. inc(p);
  359. inc(hsp);
  360. end;
  361. result[0]:=chr(hsp-@result[1]);
  362. end
  363. else
  364. result:=hs;
  365. { Strip ending / }
  366. if result[length(result)]='/' then
  367. dec(result[0]);
  368. end;
  369. function tarobjectreader.DecodeMemberSize(ahdr:TArHdr):longint;
  370. var
  371. hs : string;
  372. code : integer;
  373. hsp,
  374. p : pchar;
  375. begin
  376. p:[email protected][0];
  377. hsp:=@hs[1];
  378. while p^<>' ' do
  379. begin
  380. hsp^:=p^;
  381. inc(p);
  382. inc(hsp);
  383. end;
  384. hs[0]:=chr(hsp-@hs[1]);
  385. val(hs,result,code);
  386. if result<=0 then
  387. Comment(V_Error,'Invalid ar member size in '+filename);
  388. end;
  389. procedure tarobjectreader.ReadArchive;
  390. var
  391. currarhdr : tarhdr;
  392. nrelocs,
  393. relocidx,
  394. currfilesize,
  395. relocsize,
  396. symsize : longint;
  397. arsym : TArSymbol;
  398. s : string;
  399. syms,
  400. currp,
  401. endp,
  402. startp : pchar;
  403. relocs : plongint;
  404. begin
  405. Read(currarhdr,sizeof(currarhdr));
  406. { Read number of relocs }
  407. Read(nrelocs,sizeof(nrelocs));
  408. nrelocs:=lsb2msb(nrelocs);
  409. { Calculate sizes }
  410. currfilesize:=DecodeMemberSize(currarhdr);
  411. relocsize:=nrelocs*4;
  412. symsize:=currfilesize-relocsize-4;
  413. if symsize<0 then
  414. begin
  415. Comment(V_Error,'Illegal symtable in ar file '+filename);
  416. exit;
  417. end;
  418. { Read relocs }
  419. getmem(Relocs,relocsize);
  420. Read(relocs^,relocsize);
  421. { Read symbols, force terminating #0 to prevent overflow }
  422. getmem(syms,symsize+1);
  423. syms[symsize]:=#0;
  424. Read(syms^,symsize);
  425. { Parse symbols }
  426. relocidx:=0;
  427. currp:=syms;
  428. endp:=syms+symsize;
  429. for relocidx:=0 to nrelocs-1 do
  430. begin
  431. startp:=currp;
  432. while (currp^<>#0) do
  433. inc(currp);
  434. s[0]:=chr(currp-startp);
  435. move(startp^,s[1],byte(s[0]));
  436. arsym:=TArSymbol.create(ArSymbols,s);
  437. arsym.MemberPos:=lsb2msb(relocs[relocidx]);
  438. inc(currp);
  439. if currp>endp then
  440. begin
  441. Comment(V_Error,'Illegal symtable in ar file '+filename);
  442. break;
  443. end;
  444. end;
  445. freemem(relocs);
  446. freemem(syms);
  447. { LFN names }
  448. Read(currarhdr,sizeof(currarhdr));
  449. if DecodeMemberName(currarhdr)='/' then
  450. begin
  451. lfnsize:=DecodeMemberSize(currarhdr);
  452. getmem(lfnstrs,lfnsize);
  453. Read(lfnstrs^,lfnsize);
  454. end;
  455. end;
  456. function tarobjectreader.openfile(const fn:string):boolean;
  457. var
  458. arsym : TArSymbol;
  459. arhdr : TArHdr;
  460. begin
  461. result:=false;
  462. arsym:=TArSymbol(ArSymbols.Find(fn));
  463. if not assigned(arsym) then
  464. exit;
  465. inherited Seek(arsym.MemberPos);
  466. Read(arhdr,sizeof(arhdr));
  467. CurrMemberName:=DecodeMemberName(arhdr);
  468. CurrMemberSize:=DecodeMemberSize(arhdr);
  469. CurrMemberPos:=arsym.MemberPos+sizeof(arhdr);
  470. result:=true;
  471. end;
  472. procedure tarobjectreader.closefile;
  473. begin
  474. CurrMemberPos:=0;
  475. CurrMemberSize:=0;
  476. CurrMemberName:='';
  477. end;
  478. procedure tarobjectreader.seek(len:longint);
  479. begin
  480. inherited Seek(CurrMemberPos+len);
  481. end;
  482. end.