owar.pas 14 KB

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