StaticLinker.Mod 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068
  1. MODULE StaticLinker; (* AUTHOR "negelef"; PURPOSE "Static Object File Linker"; *)
  2. IMPORT Commands, Options, Diagnostics, Files, GenericLinker, ObjectFile, BitSets, Strings, Streams;
  3. TYPE
  4. ExportName = ARRAY 128 OF CHAR;
  5. PatchBlock*= POINTER TO RECORD
  6. baseAdr: LONGINT;
  7. addresses: LONGINT;
  8. address: ARRAY 1024 OF LONGINT;
  9. next: PatchBlock;
  10. END;
  11. RelocationInfo= OBJECT
  12. CONST blockSize=4096;
  13. VAR
  14. patchBlock: PatchBlock;
  15. PROCEDURE & Init;
  16. BEGIN
  17. patchBlock := NIL;
  18. END Init;
  19. PROCEDURE GetPatchBlock(adr: LONGINT): PatchBlock;
  20. VAR p, t: PatchBlock;
  21. BEGIN
  22. t := patchBlock;
  23. IF (patchBlock = NIL) OR (patchBlock.baseAdr > adr) THEN
  24. NEW(p); p.next := patchBlock; patchBlock := p; p.baseAdr := adr;
  25. ELSIF patchBlock.baseAdr = adr THEN p := patchBlock
  26. ELSE
  27. t := patchBlock;
  28. WHILE (t.next # NIL) & (t.next.baseAdr <= adr) DO
  29. t := t.next;
  30. END;
  31. IF t.baseAdr = adr THEN p := t
  32. ELSE
  33. NEW(p); p.next := t.next; t.next := p; p.baseAdr := adr
  34. END;
  35. END;
  36. RETURN p
  37. END GetPatchBlock;
  38. PROCEDURE AddReloc(adr: LONGINT);
  39. VAR aligned: LONGINT; p: PatchBlock;
  40. BEGIN
  41. aligned := adr - adr MOD blockSize;
  42. p := GetPatchBlock(aligned);
  43. p.address[p.addresses] := adr (*- aligned*); INC(p.addresses);
  44. END AddReloc;
  45. END RelocationInfo;
  46. ExportInfo=OBJECT
  47. VAR
  48. exports: LONGINT;
  49. name: ExportName;
  50. symbolNames: POINTER TO ARRAY OF ExportName;
  51. exportNames: POINTER TO ARRAY OF ExportName;
  52. exportAddresses: POINTER TO ARRAY OF GenericLinker.Address;
  53. PROCEDURE &Init;
  54. BEGIN
  55. exports := 0;
  56. END Init;
  57. PROCEDURE Swap(i,j: LONGINT);
  58. VAR name: ExportName; adr: LONGINT;
  59. BEGIN
  60. name := exportNames[i]; exportNames[i] := exportNames[j]; exportNames[j] := name;
  61. name := symbolNames[i]; symbolNames[i] := symbolNames[j]; symbolNames[j] := name;
  62. adr := exportAddresses[i]; exportAddresses[i] := exportAddresses[j]; exportAddresses[j] := adr;
  63. END Swap;
  64. PROCEDURE QuickSort(lo, hi: LONGINT);
  65. VAR
  66. i, j,m: LONGINT;
  67. x, t: ANY;
  68. BEGIN
  69. i := lo; j := hi; m := (lo + hi) DIV 2;
  70. WHILE i <= j DO
  71. WHILE exportNames[i] < exportNames[m] DO INC(i) END;
  72. WHILE exportNames[m] < exportNames[j] DO DEC(j) END;
  73. IF i <= j THEN
  74. IF m = i THEN m := j (* pivot will be moved by Swap *)
  75. ELSIF m = j THEN m := i (* pivot will be moved by Swap *)
  76. END;
  77. Swap(i,j);
  78. INC(i); DEC(j)
  79. END
  80. END;
  81. IF lo < j THEN QuickSort( lo, j) END;
  82. IF i < hi THEN QuickSort(i, hi) END
  83. END QuickSort;
  84. PROCEDURE Sort;
  85. BEGIN
  86. QuickSort(0, exports-1)
  87. END Sort;
  88. END ExportInfo;
  89. Arrangement* = OBJECT (GenericLinker.Arrangement);
  90. VAR
  91. displacement: GenericLinker.Address;
  92. bits: BitSets.BitSet;
  93. maxUnitSize: ObjectFile.Bits;
  94. exportInfo: ExportInfo;
  95. relocInfo: RelocationInfo;
  96. PROCEDURE & InitArrangement* (displacement: GenericLinker.Address);
  97. BEGIN
  98. SELF.displacement := displacement; maxUnitSize := 1; NEW (bits, 0); exportInfo := NIL; NEW(relocInfo);
  99. END InitArrangement;
  100. PROCEDURE Allocate* (CONST section: ObjectFile.Section): GenericLinker.Address;
  101. VAR address, alignment: ObjectFile.Bits; i: LONGINT; name: ObjectFile.SegmentedName;
  102. BEGIN
  103. IF section.unit > maxUnitSize THEN maxUnitSize := section.unit END;
  104. IF section.fixed THEN
  105. address := (section.alignment - displacement) * section.unit;
  106. ELSE
  107. address := bits.GetSize (); alignment := section.alignment * section.unit;
  108. IF alignment = 0 THEN alignment := section.unit; END;
  109. INC (address, (alignment - address MOD alignment) MOD alignment);
  110. END;
  111. IF bits.GetSize () < section.bits.GetSize () + address THEN
  112. bits.Resize (address + section.bits.GetSize ());
  113. END;
  114. (*dest pos is for some reason negative*)
  115. ASSERT(address>=0);
  116. BitSets.CopyBits (section.bits, 0, bits, address, section.bits.GetSize ());
  117. IF exportInfo # NIL THEN
  118. FOR i := 0 TO exportInfo.exports-1 DO
  119. ObjectFile.StringToSegmentedName(exportInfo.symbolNames[i], name);
  120. IF name= section.identifier.name THEN
  121. exportInfo.exportAddresses[i] := address DIV section.unit + displacement;
  122. END;
  123. END
  124. END;
  125. RETURN address DIV section.unit + displacement;
  126. END Allocate;
  127. PROCEDURE SizeInBits*(): LONGINT;
  128. BEGIN RETURN bits.GetSize()
  129. END SizeInBits;
  130. PROCEDURE Patch* (pos, value: GenericLinker.Address; offset, bits, unit: ObjectFile.Bits);
  131. BEGIN SELF.bits.SetBits ((pos - displacement) * unit + offset, bits, value);
  132. END Patch;
  133. PROCEDURE CheckReloc(target: LONGINT; pattern: ObjectFile.Pattern; CONST patch: ObjectFile.Patch);
  134. VAR i: LONGINT;
  135. BEGIN
  136. IF (pattern.mode = ObjectFile.Absolute) & (relocInfo # NIL) THEN
  137. relocInfo.AddReloc(target+patch.offset);
  138. END;
  139. END CheckReloc;
  140. END Arrangement;
  141. TYPE FileFormat = PROCEDURE (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  142. VAR defaults: Options.Defaults;
  143. processing, finished: GenericLinker.Block;
  144. PROCEDURE Align(this, to: LONGINT): LONGINT;
  145. BEGIN
  146. this := this + (-this) MOD to;
  147. RETURN this;
  148. END Align;
  149. PROCEDURE ReadObjectFile*(CONST moduleName, path, extension: ARRAY OF CHAR; linker: GenericLinker.Linker; moduleList: GenericLinker.HashTableSegmentedName);
  150. VAR fileName: Files.FileName; file: Files.File; reader: Files.Reader; i: LONGINT;
  151. offers, requires: ObjectFile.NameList;name: ARRAY 256 OF CHAR; sn: ObjectFile.SegmentedName;
  152. BEGIN
  153. linker.Information (moduleName, "processing");
  154. sn := moduleName;
  155. IF (moduleList # NIL) & (moduleList.Get(sn) = finished) THEN
  156. linker.Warning(moduleName, "duplicate");
  157. RETURN;
  158. END;
  159. IF path # "" THEN Files.JoinPath (path, moduleName, fileName); ELSE COPY(moduleName,fileName); END;
  160. file := Files.Old (fileName);
  161. IF file = NIL THEN (* try with extension *)
  162. Files.JoinExtension (fileName, extension, fileName);
  163. file := Files.Old (fileName);
  164. END;
  165. IF file = NIL THEN (* try without extension *)
  166. END;
  167. IF file = NIL THEN linker.Error (fileName, "failed to open file"); RETURN; END;
  168. Files.OpenReader (reader, file, 0);
  169. GenericLinker.OffersRequires(reader, offers, requires);
  170. reader.SetPos(0);
  171. IF moduleList # NIL THEN
  172. moduleList.Put(sn,processing);
  173. END;
  174. IF (moduleList # NIL) & (requires # NIL) THEN
  175. FOR i := 0 TO LEN(requires)-1 DO
  176. IF moduleList.Get(requires[i]) = NIL THEN
  177. name := requires[i];
  178. TRACE(name);
  179. ReadObjectFile(name,path,extension, linker, moduleList);
  180. IF linker.error THEN RETURN END;
  181. ELSIF moduleList.Get(requires[i]) = processing THEN
  182. name := requires[i];
  183. Strings.Append(name, moduleName);
  184. linker.Error(name, "cyclic import!");
  185. RETURN;
  186. END;
  187. END;
  188. END;
  189. GenericLinker.Process (reader, linker);
  190. IF moduleList # NIL THEN
  191. moduleList.Put(sn,finished);
  192. END;
  193. IF reader.res # Files.Ok THEN linker.Error (fileName, "failed to parse"); END;
  194. END ReadObjectFile;
  195. PROCEDURE WriteOutputFile* (arrangement: Arrangement; CONST fileName: Files.FileName; linker: GenericLinker.Linker; fileFormat: FileFormat);
  196. VAR file: Files.File; writer: Files.Writer; msg: ARRAY 64 OF CHAR; number: ARRAY 32 OF CHAR;
  197. BEGIN
  198. file := Files.New (fileName);
  199. Files.OpenWriter (writer, file, 0);
  200. fileFormat (linker, arrangement, writer);
  201. writer.Update; Files.Register (file);
  202. msg := "written ";
  203. Strings.IntToStr(arrangement.SizeInBits(), number);
  204. Strings.Append(msg, number);
  205. Strings.Append(msg, " bits (= ");
  206. Strings.IntToStr(arrangement.SizeInBits() DIV arrangement.maxUnitSize, number);
  207. Strings.Append(msg, number);
  208. Strings.Append(msg, " units).");
  209. linker.Information (fileName, msg);
  210. END WriteOutputFile;
  211. PROCEDURE WriteBinaryFile (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  212. VAR i: LONGINT;
  213. BEGIN
  214. FOR i := 0 TO arrangement.bits.GetSize () - 1 BY 8 DO
  215. writer.Char (CHR (arrangement.bits.GetBits (i, 8)));
  216. END;
  217. END WriteBinaryFile;
  218. PROCEDURE WriteTRMFile (arrangement: Arrangement; writer: Files.Writer; bitsPerLine, lines:LONGINT);
  219. VAR i,j,size,end,nonZeroInLeadingNibble,leadingzeros: LONGINT;
  220. PROCEDURE GetBits(pos: LONGINT): LONGINT;
  221. BEGIN
  222. IF pos >= size THEN RETURN 0
  223. ELSIF pos+4 > size THEN RETURN arrangement.bits.GetBits(pos,size-pos)
  224. ELSE RETURN arrangement.bits.GetBits(pos,4)
  225. END;
  226. END GetBits;
  227. PROCEDURE GetFirstNibble(pos: LONGINT): LONGINT;
  228. BEGIN
  229. RETURN arrangement.bits.GetBits(pos,nonZeroInLeadingNibble);
  230. END GetFirstNibble;
  231. BEGIN
  232. (*allow line size not divisible by 4 by padding the line to a multiple of 4 bits with leading zeros*)
  233. size := arrangement.bits.GetSize();
  234. end := (size-1) DIV bitsPerLine + 1;
  235. (*
  236. D.String("bits per line: ");D.Int(bitsPerLine,0);
  237. D.String(" number of bits: ");D.Int(size,0);
  238. D.String(" number of words: ");D.Int(end,0);D.Ln;
  239. *)
  240. nonZeroInLeadingNibble:=bitsPerLine MOD 4;
  241. FOR i := 0 TO end-1 DO
  242. (*be aware we are writing msb first here*)
  243. (*padded first nibble*)
  244. IF nonZeroInLeadingNibble > 0 THEN
  245. writer.Char(ObjectFile.NibbleToCharacter(GetFirstNibble(i*bitsPerLine+bitsPerLine-nonZeroInLeadingNibble) ) ) ;
  246. END;
  247. (*rest of the instruction word*)
  248. FOR j := bitsPerLine DIV 4 -1 TO 0 BY -1 DO
  249. writer.Char(ObjectFile.NibbleToCharacter(GetBits(i*bitsPerLine+j*4)));
  250. END;
  251. writer.Ln;
  252. END;
  253. IF end > 0 THEN (* allow at least one block, if size = 0 *)
  254. lines := (((end-1) DIV lines)+1)*lines; (* round up to next multiple of lines *)
  255. END;
  256. FOR i := end TO lines -1 DO
  257. FOR j:=1 TO (bitsPerLine-1) DIV 4 + 1 BY 1 DO
  258. writer.Char('F');
  259. END;
  260. writer.Ln;
  261. END;
  262. END WriteTRMFile;
  263. PROCEDURE WriteTRMCodeFile* (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  264. BEGIN
  265. WriteTRMFile (arrangement, writer, arrangement.maxUnitSize,1024);
  266. END WriteTRMCodeFile;
  267. PROCEDURE WriteTRMDataFile* (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  268. BEGIN WriteTRMFile (arrangement, writer, 32,512);
  269. END WriteTRMDataFile;
  270. PROCEDURE WritePEFile (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer; bitmode, subSystem: INTEGER; isDLL: BOOLEAN);
  271. CONST DOSText = "This program cannot be run in DOS mode.$";
  272. CONST DOSHeaderSize = 64; DOSCodeSize = 14; DOSTextSize = 40; DOSStubSize = ((DOSHeaderSize + DOSCodeSize + DOSTextSize + 15) DIV 16) * 16;
  273. CONST FileAlignment = 200H; SectionAlignment = 1000H; HeaderSize = 24; SectionHeaderSize = 40; DirectoryEntries = 16;
  274. VAR OptionalHeaderSize, CodeSize, AlignedCodeSize, HeadersSize, BaseCodeAddress, BaseAddress: LONGINT; exportInfo: ExportInfo; relocInfo: RelocationInfo;
  275. pos: LONGINT;
  276. PROCEDURE Reserve (size: LONGINT);
  277. BEGIN INC(pos, size); WHILE size # 0 DO writer.Char (0X); DEC (size); END;
  278. END Reserve;
  279. PROCEDURE WriteBYTE (value: LONGINT);
  280. BEGIN writer.Char (CHR (value)); INC(pos);
  281. END WriteBYTE;
  282. PROCEDURE WriteWORD (value: LONGINT);
  283. BEGIN WriteBYTE (value MOD 100H); WriteBYTE (value DIV 100H);
  284. END WriteWORD;
  285. PROCEDURE WriteDWORD (value: LONGINT);
  286. BEGIN WriteWORD (value MOD 10000H); WriteWORD (value DIV 10000H);
  287. END WriteDWORD;
  288. PROCEDURE WritePTR (value: LONGINT);
  289. BEGIN WriteDWORD (value); IF bitmode = 64 THEN WriteDWORD (0) END;
  290. END WritePTR;
  291. PROCEDURE WriteString(s: ARRAY OF CHAR);
  292. BEGIN
  293. writer.String(s); INC(pos, Strings.Length(s));
  294. END WriteString;
  295. PROCEDURE WriteString0(s: ARRAY OF CHAR);
  296. BEGIN
  297. WriteString(s); WriteBYTE(0);
  298. END WriteString0;
  299. PROCEDURE AlignTo(alignment: LONGINT);
  300. BEGIN alignment := Align(pos, alignment); Reserve(alignment - pos);
  301. END AlignTo;
  302. PROCEDURE ReserveTo(p: LONGINT);
  303. BEGIN ASSERT(p >= pos); Reserve(p-pos);
  304. END ReserveTo;
  305. PROCEDURE HasExports(): BOOLEAN;
  306. BEGIN
  307. RETURN (exportInfo # NIL) & (exportInfo.exports # 0)
  308. END HasExports;
  309. PROCEDURE ExportTableSize(): LONGINT;
  310. VAR i,offset: LONGINT;
  311. BEGIN
  312. IF ~HasExports() THEN RETURN 0 END;
  313. offset := 40 + Strings.Length(exportInfo.name)+1;
  314. FOR i := 0 TO exportInfo.exports-1 DO
  315. INC(offset, Strings.Length(exportInfo.exportNames[i])+1);
  316. INC(offset, 10);
  317. END;
  318. RETURN offset
  319. END ExportTableSize;
  320. PROCEDURE AlignedExportTableSize(): LONGINT;
  321. BEGIN
  322. RETURN Align(ExportTableSize(), SectionAlignment);
  323. END AlignedExportTableSize;
  324. PROCEDURE HasRelocs(): BOOLEAN;
  325. BEGIN
  326. RETURN (relocInfo # NIL) & (relocInfo.patchBlock # NIL)
  327. END HasRelocs;
  328. PROCEDURE RelocTableSize(): LONGINT;
  329. VAR p: PatchBlock; size: LONGINT;
  330. BEGIN
  331. IF ~HasRelocs() THEN RETURN 0 END;
  332. size := 0;
  333. p := relocInfo.patchBlock;
  334. WHILE p # NIL DO
  335. INC(size, 8 + p.addresses * 2);
  336. IF ODD(p.addresses) THEN INC(size, 2) END;
  337. p := p.next
  338. END;
  339. RETURN size
  340. END RelocTableSize;
  341. PROCEDURE AlignedRelocTableSize(): LONGINT;
  342. BEGIN RETURN Align(RelocTableSize(), SectionAlignment);
  343. END AlignedRelocTableSize;
  344. PROCEDURE SectionHeaderOffset(): LONGINT;
  345. BEGIN RETURN DOSStubSize + HeaderSize + OptionalHeaderSize
  346. END SectionHeaderOffset;
  347. PROCEDURE WriteDOSStub;
  348. BEGIN
  349. WriteWORD (5A4DH); (* e_magic *)
  350. WriteWORD (DOSStubSize); (* e_cblp *)
  351. WriteWORD (1); (* e_cp *)
  352. WriteWORD (0); (* e_crlc *)
  353. WriteWORD (DOSHeaderSize DIV 16); (* e_cparhdr *)
  354. WriteWORD (0); (* e_minalloc *)
  355. WriteWORD (0); (* e_maxalloc *)
  356. WriteWORD (0); (* e_ss *)
  357. WriteWORD (0); (* e_sp *)
  358. WriteWORD (0); (* e_csum *)
  359. WriteWORD (0); (* e_ip *)
  360. WriteWORD (0); (* e_cs *)
  361. WriteWORD (DOSHeaderSize); (* e_lfarlc *)
  362. WriteWORD (0); (* e_ovno *)
  363. Reserve (32); (* e_res *)
  364. WriteDWORD (DOSStubSize); (* e_lfanew *)
  365. WriteBYTE (00EH); WriteBYTE (01FH); WriteBYTE (0BAH); WriteWORD (DOSCodeSize);
  366. WriteBYTE (0B4H); WriteBYTE (009H); WriteBYTE (0CDH); WriteBYTE (021H); WriteBYTE (0B8H);
  367. WriteBYTE (001H); WriteBYTE (04CH); WriteBYTE (0CDH); WriteBYTE (021H); WriteString (DOSText);
  368. Reserve (DOSStubSize - DOSHeaderSize - DOSCodeSize - DOSTextSize);
  369. END WriteDOSStub;
  370. PROCEDURE WriteHeader;
  371. VAR characteristics, sections: LONGINT;
  372. BEGIN
  373. WriteDWORD (000004550H); (* Signature *)
  374. IF bitmode = 64 THEN
  375. WriteWORD (08664H); (* Machine *)
  376. ELSE
  377. WriteWORD (0014CH); (* Machine *)
  378. END;
  379. sections := 1;
  380. IF HasRelocs() THEN INC(sections) END;
  381. IF HasExports()THEN INC(sections) END;
  382. WriteWORD (sections); (* NumberOfSections *)
  383. WriteDWORD (0); (* TimeDateStamp *)
  384. WriteDWORD (0); (* PointerToSymbolTable *)
  385. WriteDWORD (0); (* NumberOfSymbols *)
  386. WriteWORD (OptionalHeaderSize); (* SizeOfOptionalHeader *)
  387. characteristics := 222H;
  388. IF ~HasRelocs() THEN characteristics := characteristics + 1 END;
  389. IF isDLL THEN characteristics := characteristics + 2000H END;
  390. IF bitmode#64 THEN characteristics := characteristics + 100H END;
  391. WriteWORD(characteristics);
  392. END WriteHeader;
  393. PROCEDURE WriteOptionalHeader;
  394. VAR ImageSize: LONGINT;
  395. BEGIN
  396. ImageSize := Align(BaseCodeAddress+AlignedCodeSize+AlignedExportTableSize()+AlignedRelocTableSize(), SectionAlignment);
  397. (* 0 *) IF bitmode = 64 THEN
  398. WriteWORD (0020BH); (* Magic *)
  399. ELSE
  400. WriteWORD (0010BH); (* Magic *)
  401. END;
  402. (* 2 *) WriteBYTE (0); (* MajorLinkerVersion *)
  403. (* 3 *) WriteBYTE (0); (* MinorLinkerVersion *)
  404. (* 4 *) WriteDWORD (CodeSize); (* SizeOfCode *)
  405. (* 8 *) WriteDWORD (0); (* SizeOfInitializedData *)
  406. (* 12 *) WriteDWORD (0); (* SizeOfUninitializedData *)
  407. (* 16 *) WriteDWORD (BaseCodeAddress); (* AddressOfEntryPoint *)
  408. (* 20 *) WriteDWORD (BaseCodeAddress); (* BaseOfCode *)
  409. (* 24 *) IF bitmode # 64 THEN
  410. WriteDWORD (ImageSize); (* BaseOfData *)
  411. END;
  412. (* 28 / 24 *) WritePTR (arrangement.displacement - BaseCodeAddress); (* ImageBase *)
  413. (* 32 *) WriteDWORD (SectionAlignment); (* SectionAlignment *)
  414. (* 36 *)WriteDWORD (FileAlignment); (* FileAlignment *)
  415. (* 40 *)WriteWORD (4); (* MajorOperatingSystemVersion *)
  416. (* 42 *)WriteWORD (0); (* MinorOperatingSystemVersion *)
  417. (* 44 *)WriteWORD (0); (* MajorImageVersion *)
  418. (* 46 *)WriteWORD (0); (* MinorImageVersion *)
  419. (* 48 *)WriteWORD (4); (* MajorSubsystemVersion *)
  420. (* 50 *)WriteWORD (0); (* MinorSubsystemVersion *)
  421. (* 52 *)WriteDWORD (0); (* Win32VersionValue *)
  422. (* 56 *)WriteDWORD (ImageSize); (* SizeOfImage *)
  423. (* 60 *)WriteDWORD (HeadersSize); (* SizeOfHeaders *)
  424. (* 64 *)WriteDWORD (0); (* CheckSum *)
  425. (* 68 *)WriteWORD (subSystem); (* Subsystem *)
  426. (* 70 *)IF isDLL THEN WriteWORD (40H); (* DllCharacteristics *)
  427. ELSE WriteWORD(0)
  428. END;
  429. (* 72 *)WritePTR (0100000H); (* SizeOfStackReserve *)
  430. (* 76 / 80 *)WritePTR (01000H); (* SizeOfStackCommit *)
  431. (* 80 / 88 *)WritePTR (0100000H); (* SizeOfHeapReserve *)
  432. (* 84 / 96 *)WritePTR (01000H); (* SizeOfHeapCommit *)
  433. (* 88 / 104 *)WriteDWORD (0); (* LoaderFlags *)
  434. (* 92 / 108 *)WriteDWORD (DirectoryEntries); (* NumberOfRvaAndSizes *)
  435. IF HasExports() THEN
  436. (* 96 / 112 *) WriteDWORD(BaseCodeAddress + AlignedCodeSize); WriteDWORD(ExportTableSize()); (* location and size of export table *)
  437. ELSE Reserve(8)
  438. END;
  439. (* 104 / 120 *) WriteDWORD (BaseCodeAddress +8) ; WriteDWORD (40); (* location and size of of idata section *)
  440. Reserve (3 * 8);
  441. IF HasRelocs() THEN
  442. WriteDWORD (BaseCodeAddress + AlignedCodeSize+AlignedExportTableSize()) ; WriteDWORD (RelocTableSize()); (* location and size of of reloc section *)
  443. ELSE Reserve(8)
  444. END;
  445. Reserve ((DirectoryEntries - 6) * 8);
  446. END WriteOptionalHeader;
  447. PROCEDURE WriteSections;
  448. VAR ExportNameTableAddress, i, offset: LONGINT; p: PatchBlock;
  449. BEGIN
  450. (* code section header *)
  451. (* 0 *) WriteString (".text"); Reserve (3); (* Name *)
  452. (* 8 *) WriteDWORD (CodeSize); (* VirtualSize *)
  453. (* 12 *) WriteDWORD (BaseCodeAddress); (* VirtualAddress *)
  454. (* 16 *) WriteDWORD (AlignedCodeSize); (* SizeOfRawData *)
  455. (* 20 *) WriteDWORD (BaseCodeAddress); (* PointerToRawData *)
  456. (* 24 *) WriteDWORD (0); (* PointerToRelocations *)
  457. (* 28 *) WriteDWORD (0); (* PointerToLinenumbers *)
  458. (* 32 *) WriteWORD (0); (* NumberOfRelocations *)
  459. (* 34 *) WriteWORD (0); (* NumberOfLinenumbers *)
  460. (* 36 *) WriteDWORD (LONGINT (0E0000020H)); (* Characteristics *)
  461. IF HasExports() THEN
  462. (* export table header *)
  463. (* 0 *) WriteString(".edata"); Reserve(2); (* name *)
  464. (* 8 *) WriteDWORD(ExportTableSize()); (* virtual size *)
  465. (* 12 *) WriteDWORD(BaseCodeAddress + AlignedCodeSize (* address *));
  466. (* 16 *) WriteDWORD(AlignedExportTableSize()); (* raw data size *)
  467. (* 20 *) WriteDWORD(BaseCodeAddress + AlignedCodeSize); (* raw data pointer *)
  468. (* 24 *) WriteDWORD (0); (* PointerToRelocations *)
  469. (* 28 *) WriteDWORD (0); (* PointerToLinenumbers *)
  470. (* 32 *) WriteWORD (0); (* NumberOfRelocations *)
  471. (* 34 *) WriteWORD (0); (* NumberOfLinenumbers *)
  472. (* 36 *) WriteDWORD (LONGINT (040000000H)); (* Characteristics *)
  473. END;
  474. IF HasRelocs() THEN
  475. (* reloc table header *)
  476. (* 0 *) WriteString(".reloc"); Reserve(2); (* name *)
  477. (* 8 *) WriteDWORD(RelocTableSize()); (* virtual size *)
  478. (* 12 *) WriteDWORD(BaseCodeAddress + AlignedCodeSize + AlignedExportTableSize() (* address *));
  479. (* 16 *) WriteDWORD(AlignedRelocTableSize()); (* raw data size *)
  480. (* 20 *) WriteDWORD(BaseCodeAddress + AlignedCodeSize+ AlignedExportTableSize()); (* raw data pointer *)
  481. (* 24 *) WriteDWORD (0); (* PointerToRelocations *)
  482. (* 28 *) WriteDWORD (0); (* PointerToLinenumbers *)
  483. (* 32 *) WriteWORD (0); (* NumberOfRelocations *)
  484. (* 34 *) WriteWORD (0); (* NumberOfLinenumbers *)
  485. (* 36 *) WriteDWORD (LONGINT (040000000H)); (* Characteristics *)
  486. END;
  487. (* 40 / 80 / 120 *) ReserveTo(BaseCodeAddress);
  488. (**** code section *****)
  489. (* BaseCodeAddress *)
  490. WriteBinaryFile (linker, arrangement, writer); INC(pos, arrangement.bits.GetSize () DIV 8);
  491. (* BaseCodeAddress +CodeSize *)
  492. Reserve (AlignedCodeSize-CodeSize);
  493. (* BaseCodeAddress + AlignedCodeSize *)
  494. IF HasExports() THEN
  495. (***** export section *****)
  496. (* BaseCodeAddress + AlignedCodeSize *)
  497. (* 0 *) WriteDWORD(0); (* reserved *)
  498. (* 4 *) WriteDWORD(0); (* time / date *)
  499. (* 6 *) WriteWORD(0); (* major version *)
  500. (* 8 *) WriteWORD(0); (* minor version *)
  501. (* 12 *) WriteDWORD(BaseCodeAddress+AlignedCodeSize + 40 + 10* exportInfo.exports); (* RVA of DLL name *)
  502. (* 16 *) WriteDWORD(1); (* start ordinal number *)
  503. (* 20 *) WriteDWORD(exportInfo.exports); (* number of entries in export table *)
  504. (* 24 *) WriteDWORD(exportInfo.exports); (* number of entries in name pointer table *)
  505. (* 28 *) WriteDWORD(BaseCodeAddress+AlignedCodeSize + 40 ); (* export address table RVA *)
  506. (* 32 *) WriteDWORD(BaseCodeAddress+AlignedCodeSize + 40 + 4* exportInfo.exports); (* name pointer RVA *)
  507. (* 36 *) WriteDWORD(BaseCodeAddress+AlignedCodeSize + 40 + 8* exportInfo.exports); (* ordinal table RVA *)
  508. (* 40 *)
  509. (* export address table *)
  510. FOR i := 0 TO exportInfo.exports-1 DO
  511. ASSERT(exportInfo.exportAddresses[i] # 0);
  512. WriteDWORD(exportInfo.exportAddresses[i]-BaseAddress+BaseCodeAddress); (* RVA ! *)
  513. END;
  514. (* export name pointer table *)
  515. (* 40 + 4 * Number exports *)
  516. ExportNameTableAddress := BaseCodeAddress + AlignedCodeSize + 40 + 10* exportInfo.exports ;
  517. offset := Strings.Length(exportInfo.name)+1;
  518. FOR i := 0 TO exportInfo.exports-1 DO
  519. WriteDWORD(ExportNameTableAddress + offset);
  520. INC(offset, Strings.Length(exportInfo.exportNames[i])+1);
  521. END;
  522. (* export ordinal table *)
  523. (* 40 + 8* NumberExports *)
  524. FOR i := 0 TO exportInfo.exports-1 DO
  525. WriteWORD(i);
  526. END;
  527. (* 40 + 10* NumberExports *)
  528. (* export name table *)
  529. WriteString0(exportInfo.name);
  530. FOR i := 0 TO exportInfo.exports-1 DO
  531. WriteString0(exportInfo.exportNames[i]); (* 0x terminated *)
  532. END;
  533. Reserve(AlignedExportTableSize() - ExportTableSize());
  534. END;
  535. IF HasRelocs() THEN
  536. p := relocInfo.patchBlock;
  537. WHILE p # NIL DO
  538. WriteDWORD(p.baseAdr -BaseAddress+BaseCodeAddress) (* RVA of block *);
  539. WriteDWORD(8+p.addresses*2+2*(p.addresses MOD 2)); (* number bytes of this block *)
  540. FOR i := 0 TO p.addresses-1 DO
  541. WriteWORD(p.address[i] - p.baseAdr+ 3000H);(* block-relative addresses, highlow *)
  542. END;
  543. AlignTo(4);
  544. p := p.next;
  545. END;
  546. Reserve(AlignedRelocTableSize() - RelocTableSize());
  547. END;
  548. END WriteSections;
  549. BEGIN
  550. pos := 0;
  551. exportInfo := arrangement.exportInfo;
  552. relocInfo := arrangement.relocInfo;
  553. IF HasExports() THEN exportInfo.Sort END;
  554. BaseAddress := arrangement.displacement; (* ASSERT (arrangement.displacement = BaseAddress); *)
  555. OptionalHeaderSize := 96 + DirectoryEntries * 8;
  556. IF bitmode = 64 THEN INC (OptionalHeaderSize, 16); END;
  557. CodeSize := arrangement.bits.GetSize () DIV 8;
  558. AlignedCodeSize := Align(CodeSize, SectionAlignment);
  559. HeadersSize := Align(DOSStubSize + HeaderSize + OptionalHeaderSize + SectionHeaderSize, FileAlignment);
  560. BaseCodeAddress := Align(HeadersSize, SectionAlignment);
  561. WriteDOSStub; WriteHeader; WriteOptionalHeader; WriteSections;
  562. END WritePEFile;
  563. PROCEDURE WriteDLL32File (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  564. BEGIN WritePEFile (linker, arrangement, writer, 32, 2, TRUE);
  565. END WriteDLL32File;
  566. PROCEDURE WriteDLL64File (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  567. BEGIN WritePEFile (linker, arrangement, writer, 64, 2, TRUE);
  568. END WriteDLL64File;
  569. PROCEDURE WritePE32File (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  570. BEGIN WritePEFile (linker, arrangement, writer, 32, 2, FALSE);
  571. END WritePE32File;
  572. PROCEDURE WritePE64File (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  573. BEGIN WritePEFile (linker, arrangement, writer, 64, 2, FALSE);
  574. END WritePE64File;
  575. PROCEDURE WritePE32CUIFile (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  576. BEGIN WritePEFile (linker, arrangement, writer, 32, 3, FALSE);
  577. END WritePE32CUIFile;
  578. PROCEDURE WritePE64CUIFile (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  579. BEGIN WritePEFile (linker, arrangement, writer, 64, 3, FALSE);
  580. END WritePE64CUIFile;
  581. PROCEDURE WriteEFI32File (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  582. BEGIN WritePEFile (linker, arrangement, writer, 32, 10, FALSE);
  583. END WriteEFI32File;
  584. PROCEDURE WriteEFI64File (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  585. BEGIN WritePEFile (linker, arrangement, writer, 64, 10, FALSE);
  586. END WriteEFI64File;
  587. PROCEDURE WriteELFFile (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  588. CONST ELFHeaderSize = 52; ProgramHeaderSize = 32; HeadersSize = ELFHeaderSize + ProgramHeaderSize;
  589. CONST BaseAddress = 08048000H; EntryAddress = BaseAddress + HeadersSize;
  590. PROCEDURE Reserve (size: LONGINT);
  591. BEGIN WHILE size # 0 DO writer.Char (0X); DEC (size); END;
  592. END Reserve;
  593. PROCEDURE WriteByte (value: LONGINT);
  594. BEGIN writer.Char (CHR (value));
  595. END WriteByte;
  596. PROCEDURE WriteHalf (value: LONGINT);
  597. BEGIN WriteByte (value MOD 100H); WriteByte (value DIV 100H);
  598. END WriteHalf;
  599. PROCEDURE WriteWord (value: LONGINT);
  600. BEGIN WriteHalf (value MOD 10000H); WriteHalf (value DIV 10000H);
  601. END WriteWord;
  602. PROCEDURE WriteELFHeader;
  603. BEGIN
  604. WriteByte (7FH); (* e_ident[EI_MAG0] *)
  605. WriteByte (ORD('E')); (* e_ident[EI_MAG1] *)
  606. WriteByte (ORD('L')); (* e_ident[EI_MAG2] *)
  607. WriteByte (ORD('F')); (* e_ident[EI_MAG3] *)
  608. WriteByte (1); (* e_ident[EI_CLASS] *)
  609. WriteByte (1); (* e_ident[EI_DATA] *)
  610. WriteByte (1); (* e_ident[EI_VERSION] *)
  611. WriteByte (0); (* e_ident[EI_PAD] *)
  612. Reserve (8); (* e_ident[EI_NIDENT] *)
  613. WriteHalf (2); (* e_type *)
  614. WriteHalf (3); (* e_machine *)
  615. WriteWord (1); (* e_version *)
  616. WriteWord (EntryAddress); (* e_entry *)
  617. WriteWord (ELFHeaderSize); (* e_phoff *)
  618. WriteWord (0); (* e_phoff *)
  619. WriteWord (0); (* e_flags *)
  620. WriteHalf (ELFHeaderSize); (* e_ehsize *)
  621. WriteHalf (ProgramHeaderSize); (* e_phentsize *)
  622. WriteHalf (1); (* e_phnum *)
  623. WriteHalf (0); (* e_shentsize *)
  624. WriteHalf (0); (* e_shnum *)
  625. WriteHalf (0); (* e_shstrndx *)
  626. END WriteELFHeader;
  627. PROCEDURE WriteProgramHeader;
  628. VAR FileSize: LONGINT;
  629. BEGIN
  630. FileSize := HeadersSize + arrangement.bits.GetSize () DIV 8;
  631. WriteWord (1); (* p_type *)
  632. WriteWord (0); (* p_offset *)
  633. WriteWord (BaseAddress); (* p_vaddr *)
  634. WriteWord (0); (* p_paddr *)
  635. WriteWord (FileSize); (* p_filesz *)
  636. WriteWord (FileSize); (* p_memsz *)
  637. WriteWord (7); (* p_flags *)
  638. WriteWord (1000H); (* p_align *)
  639. END WriteProgramHeader;
  640. BEGIN
  641. ASSERT (arrangement.displacement = BaseAddress);
  642. WriteELFHeader;
  643. WriteProgramHeader;
  644. WriteBinaryFile (linker, arrangement, writer);
  645. END WriteELFFile;
  646. PROCEDURE WriteMachOFile (linker: GenericLinker.Linker; arrangement: Arrangement; writer: Files.Writer);
  647. CONST SegmentName = "__TEXT"; SectionName = "__text";
  648. CONST MachHeaderSize = 28; LoadCommandSize = 124; ThreadCommandSize = 80;
  649. CONST CommandsSize = LoadCommandSize + ThreadCommandSize; Start = MachHeaderSize + CommandsSize;
  650. CONST BaseAddress = 000010E8H;
  651. PROCEDURE Write (value: LONGINT);
  652. BEGIN writer.Char (CHR (value)); writer.Char (CHR (value DIV 100H)); writer.Char (CHR (value DIV 10000H)); writer.Char (CHR (value DIV 1000000H));
  653. END Write;
  654. PROCEDURE WriteName (CONST name: ARRAY OF CHAR);
  655. VAR i: INTEGER;
  656. BEGIN i := 0; WHILE name[i] # 0X DO writer.Char (name[i]); INC (i); END;
  657. WHILE i # 16 DO writer.Char (0X); INC (i); END;
  658. END WriteName;
  659. PROCEDURE WriteMachHeader;
  660. BEGIN
  661. Write (LONGINT (0FEEDFACEH)); (* magic *)
  662. Write (7); (* cputype *)
  663. Write (3); (* cpusubtype *)
  664. Write (2); (* filetype *)
  665. Write (2); (* ncmds *)
  666. Write (CommandsSize); (* sizeofcmds *)
  667. Write (0); (* flags *)
  668. END WriteMachHeader;
  669. PROCEDURE WriteLoadCommand;
  670. VAR FileSize: LONGINT;
  671. BEGIN
  672. FileSize := MachHeaderSize + CommandsSize + arrangement.bits.GetSize () DIV 8;
  673. Write (1); (* cmd *)
  674. Write (LoadCommandSize); (* cmdsize *)
  675. WriteName (SegmentName); (* segname *)
  676. Write (BaseAddress - Start); (* vmaddr *)
  677. Write (FileSize); (* vmsize *)
  678. Write (0); (* fileoff *)
  679. Write (FileSize); (* filesize *)
  680. Write (7); (* maxprot *)
  681. Write (7); (* initprot *)
  682. Write (1); (* nsects *)
  683. Write (0); (* flags *)
  684. WriteName (SectionName); (* sectname *)
  685. WriteName (SegmentName); (* segname *)
  686. Write (BaseAddress); (* addr *)
  687. Write (arrangement.bits.GetSize () DIV 8); (* size *)
  688. Write (Start); (* offset *)
  689. Write (2); (* align *)
  690. Write (0); (* reloff *)
  691. Write (0); (* nreloc *)
  692. Write (0); (* flags *)
  693. Write (0); (* reserved1 *)
  694. Write (0); (* reserved2 *)
  695. END WriteLoadCommand;
  696. PROCEDURE WriteThreadCommand;
  697. BEGIN
  698. Write (5); (* cmd *)
  699. Write (ThreadCommandSize); (* cmdsize *)
  700. Write (1); (* flavor *)
  701. Write (16); (* count *)
  702. Write (0); (* eax *)
  703. Write (0); (* ebx *)
  704. Write (0); (* ecx *)
  705. Write (0); (* edx *)
  706. Write (0); (* edi *)
  707. Write (0); (* esi *)
  708. Write (0); (* ebp *)
  709. Write (0); (* esp *)
  710. Write (0); (* ss *)
  711. Write (0); (* eflags *)
  712. Write (BaseAddress); (* eip *)
  713. Write (0); (* cs *)
  714. Write (0); (* ds *)
  715. Write (0); (* es *)
  716. Write (0); (* fs *)
  717. Write (0); (* gs *)
  718. END WriteThreadCommand;
  719. BEGIN
  720. ASSERT (arrangement.displacement = BaseAddress);
  721. WriteMachHeader;
  722. WriteLoadCommand;
  723. WriteThreadCommand;
  724. WriteBinaryFile (linker, arrangement, writer);
  725. END WriteMachOFile;
  726. PROCEDURE GetFileFormat (options: Options.Options; CONST name: Options.Name; default: FileFormat): FileFormat;
  727. VAR format: ARRAY 10 OF CHAR;
  728. BEGIN
  729. IF ~options.GetString (name, format) THEN RETURN default;
  730. ELSIF format = "TRMCode" THEN RETURN WriteTRMCodeFile;
  731. ELSIF format = "TRMData" THEN RETURN WriteTRMDataFile;
  732. ELSIF format = "PE32" THEN RETURN WritePE32File;
  733. ELSIF format = "PE64" THEN RETURN WritePE64File;
  734. ELSIF format = "PE32CUI" THEN RETURN WritePE32CUIFile;
  735. ELSIF format = "PE64CUI" THEN RETURN WritePE64CUIFile;
  736. ELSIF format = "EFI32" THEN RETURN WriteEFI32File;
  737. ELSIF format = "EFI64" THEN RETURN WriteEFI64File;
  738. ELSIF format = "ELF" THEN RETURN WriteELFFile;
  739. ELSIF format = "MACHO" THEN RETURN WriteMachOFile;
  740. ELSIF format = "DLL32" THEN RETURN WriteDLL32File;
  741. ELSIF format = "DLL64" THEN RETURN WriteDLL64File;
  742. ELSE RETURN default; END;
  743. END GetFileFormat;
  744. PROCEDURE ParseExports(CONST names: ARRAY OF CHAR): ExportInfo;
  745. VAR number: LONGINT; info: ExportInfo; pos: LONGINT; name: ExportName;
  746. PROCEDURE SkipWhitespace;
  747. BEGIN WHILE (names[pos] # 0X) & (names[pos] <= " ") DO INC(pos) END;
  748. END SkipWhitespace;
  749. PROCEDURE ReadName(VAR name: ARRAY OF CHAR): BOOLEAN;
  750. VAR i: LONGINT;
  751. BEGIN
  752. i := 0;
  753. WHILE (names[pos] # 0X) & (names[pos] > " ") & (names[pos] # ",") & (names[pos] # "=") DO name[i] := names[pos]; INC(i); INC(pos) END;
  754. name[i] := 0X;
  755. RETURN i > 0;
  756. END ReadName;
  757. PROCEDURE ParseEntry(VAR symbolName, exportName: ARRAY OF CHAR): BOOLEAN;
  758. BEGIN
  759. SkipWhitespace;
  760. IF ReadName(symbolName) THEN
  761. SkipWhitespace;
  762. IF names[pos] = "=" THEN
  763. INC(pos); SkipWhitespace;
  764. IF ~ReadName(exportName) THEN RETURN FALSE END;
  765. SkipWhitespace;
  766. ELSE COPY(symbolName, exportName);
  767. END;
  768. IF names[pos] = "," THEN INC(pos); SkipWhitespace END;
  769. RETURN TRUE
  770. ELSE RETURN FALSE
  771. END;
  772. END ParseEntry;
  773. BEGIN
  774. pos := 0; number := 0;
  775. WHILE ParseEntry(name, name) DO INC(number) END;
  776. IF (names[pos] # 0X) OR (number = 0) THEN RETURN NIL END;
  777. NEW(info);
  778. NEW(info.symbolNames, number);
  779. NEW(info.exportNames, number);
  780. NEW(info.exportAddresses, number);
  781. info.exports := number;
  782. number := 0; pos := 0;
  783. WHILE (names[pos] # 0X) & ParseEntry(info.symbolNames[number], info.exportNames[number]) DO
  784. INC(number)
  785. END;
  786. RETURN info
  787. END ParseExports;
  788. PROCEDURE CheckExports(info: ExportInfo; error: Streams.Writer): BOOLEAN;
  789. VAR success: BOOLEAN; i: LONGINT;
  790. BEGIN
  791. IF info = NIL THEN RETURN TRUE END;
  792. success := TRUE;
  793. FOR i := 0 TO info.exports-1 DO
  794. IF info.exportAddresses[i] = 0 THEN
  795. error.String("Export symbol not found: "); error.String(info.symbolNames[i]); error.String(" (= ");
  796. error.String(info.exportNames[i]); error.String(")"); error.Ln;
  797. success := FALSE;
  798. END;
  799. END;
  800. RETURN success
  801. END CheckExports;
  802. PROCEDURE Link* (context: Commands.Context);
  803. VAR options: Options.Options;
  804. silent, useAll, strict: BOOLEAN;
  805. codeFileFormat, dataFileFormat: FileFormat;
  806. codeDisplacement, dataDisplacement: GenericLinker.Address;
  807. path, extension, codeFileName, dataFileName, moduleName, logFileName, tempName: Files.FileName;
  808. diagnostics: Diagnostics.StreamDiagnostics; code, data: Arrangement; linker: GenericLinker.Linker;
  809. linkRoot: ARRAY 256 OF CHAR; logFile: Files.File; log: Files.Writer;
  810. use: SET;
  811. exportString: ARRAY 1024 OF CHAR;
  812. error: BOOLEAN;
  813. parsed: BOOLEAN;
  814. position: LONGINT;
  815. name: ARRAY 32 OF CHAR;
  816. reader: Streams.Reader;
  817. moduleList: GenericLinker.HashTableSegmentedName;
  818. segmentedName: ObjectFile.SegmentedName;
  819. BEGIN
  820. NEW (options);
  821. options.Add('p',"platform", Options.String);
  822. options.Add (0X, "silent", Options.Flag);
  823. options.Add ('a', "useAll", Options.Flag);
  824. options.Add ('s', "strict", Options.Flag);
  825. options.Add (0X, "path", Options.String); options.Add (0X, "extension", Options.String);
  826. options.Add (0X, "fileName", Options.String); options.Add (0X, "dataFileName", Options.String);
  827. options.Add (0X, "displacement", Options.Integer); options.Add (0X, "dataDisplacement", Options.Integer);
  828. options.Add (0X, "fileFormat", Options.String); options.Add (0X, "dataFileFormat", Options.String);
  829. options.Add (0X, "logFileName", Options.String);
  830. options.Add(0X,"linkRoot", Options.String);
  831. options.Add(0X,"exports", Options.String);
  832. position := context.arg.Pos();
  833. parsed := options.Parse(context.arg,NIL);
  834. IF options.GetString("platform", name) THEN
  835. reader := defaults.Get(name);
  836. IF reader = NIL THEN
  837. context.error.String("Unknown platform"); context.error.Ln
  838. ELSE
  839. parsed := options.Parse(reader, NIL) & parsed;
  840. context.arg.SetPos(position);
  841. context.arg.SetPos(position);
  842. parsed := options.Parse(context.arg, context.error) & parsed; (* reparse overwrites *)
  843. END;
  844. END;
  845. IF ~parsed THEN context.result := Commands.CommandParseError; RETURN; END;
  846. silent := options.GetFlag ("silent");
  847. useAll := options.GetFlag ("useAll");
  848. strict := options.GetFlag ("strict");
  849. IF ~options.GetString ("path", path) THEN path := ""; END;
  850. IF ~options.GetString ("extension", extension) THEN extension := ObjectFile.DefaultExtension; END;
  851. IF ~options.GetString ("fileName", codeFileName) THEN codeFileName := "linker.bin"; END;
  852. IF ~options.GetString ("dataFileName", dataFileName) THEN dataFileName := codeFileName; END;
  853. IF ~options.GetString ("logFileName", logFileName) THEN
  854. COPY(codeFileName, logFileName); Files.SplitExtension(logFileName,logFileName,tempName); Files.JoinExtension(logFileName,"log",logFileName);
  855. END;
  856. IF ~options.GetInteger ("displacement", codeDisplacement) THEN codeDisplacement := 0; END;
  857. IF ~options.GetInteger ("dataDisplacement", codeDisplacement) THEN dataDisplacement := codeDisplacement; END;
  858. codeFileFormat := GetFileFormat (options, "fileFormat", WriteBinaryFile);
  859. dataFileFormat := GetFileFormat (options, "dataFileFormat", codeFileFormat);
  860. NEW (code, codeDisplacement);
  861. IF options.GetString("exports", exportString) THEN
  862. code.exportInfo := ParseExports(exportString);
  863. IF code.exportInfo = NIL THEN
  864. context.error.String("Syntax error in export list or empty export list."); context.error.Ln;
  865. ELSE
  866. COPY(codeFileName, code.exportInfo.name);
  867. END;
  868. END;
  869. IF codeFileName # dataFileName THEN NEW (data, dataDisplacement); ELSE data := code; END;
  870. NEW (diagnostics, context.error);
  871. logFile := Files.New(logFileName);
  872. IF logFile # NIL THEN NEW(log, logFile,0) ELSE log := NIL END;
  873. IF useAll THEN use := GenericLinker.UseAll ELSE use := GenericLinker.UseInitCode END;
  874. NEW (linker, diagnostics, log, use, code, data);
  875. IF options.GetString("linkRoot",linkRoot) THEN linker.SetLinkRoot(linkRoot) END;
  876. NEW(moduleList,32);
  877. segmentedName := "SYSTEM";
  878. moduleList.Put(segmentedName, finished);
  879. segmentedName := "system";
  880. moduleList.Put(segmentedName, finished);
  881. WHILE ~linker.error & context.arg.GetString (moduleName) DO
  882. ReadObjectFile (moduleName, path, extension, linker, moduleList);
  883. IF strict & ~linker.error THEN linker.Resolve END;
  884. END;
  885. (* do linking after having read in all blocks to account for potential constraints *)
  886. IF ~linker.error THEN linker.Link; END;
  887. error := linker.error;
  888. error := error OR ~CheckExports(code.exportInfo, context.error);
  889. IF ~error THEN
  890. IF (code.displacement # 0) & (linker.log # NIL) THEN linker.log.String("code displacement 0"); linker.log.Hex(code.displacement,-8); linker.log.String("H"); linker.log.Ln END;
  891. WriteOutputFile (code, codeFileName, linker, codeFileFormat);
  892. IF data # code THEN
  893. IF (data.displacement # 0) & (linker.log # NIL) THEN linker.log.String("data displacement 0"); linker.log.Hex(data.displacement,-8); linker.log.String("H"); linker.log.Ln END;
  894. WriteOutputFile (data, dataFileName, linker, dataFileFormat);
  895. END;
  896. IF ~silent THEN
  897. context.out.String("Link successful. Written files: ");
  898. context.out.String(codeFileName);
  899. IF data # code THEN context.out.String(", "); context.out.String(dataFileName) END;
  900. IF logFile # NIL THEN context.out.String(", "); context.out.String(logFileName); END;
  901. context.out.Ln
  902. END;
  903. IF log # NIL THEN
  904. log.Update; Files.Register(logFile);
  905. END;
  906. END;
  907. IF error THEN context.result := Commands.CommandError; END;
  908. END Link;
  909. PROCEDURE ShowDefaults*(context: Commands.Context);
  910. BEGIN
  911. defaults.Show(context.out);
  912. END ShowDefaults;
  913. BEGIN
  914. NEW(processing);
  915. NEW(finished);
  916. NEW(defaults);
  917. defaults.Add("Win32C","--fileFormat=PE32CUI --fileName=oberon.exe --extension=GofW --displacement=401000H");
  918. defaults.Add("Linux32G","--fileFormat=Raw --fileName=oberon --extension=.GofU --displacement=08048000H");
  919. defaults.Add("RPI","--fileName=kernel.img --extension=.Goa --displacement=8000H -sa ");
  920. END StaticLinker.
  921. StaticLinker.Link --fileName=test.exe --fileFormat=PE32 --displacement=401000H Test ~
  922. StaticLinker.Link --fileName=a.out --fileFormat=ELF --displacement=08048000H Test ~
  923. StaticLinker.Link --fileName=a.out --fileFormat=MACHO --displacement=000010E8H Test ~