Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1373 lines
28 KiB

  1. /*++
  2. Copyright (c) 2000 Microsoft Corporation
  3. Module Name:
  4. bldrthnk.c
  5. Abstract:
  6. This module implements a program which generates code to thunk from
  7. 32-bit to 64-bit structures.
  8. This code is generated as an aid to the AMD64 boot loader, which
  9. must generate 64-bit structures from 32-bit structures.
  10. Author:
  11. Forrest C. Foltz (forrestf) 15-May-2000
  12. To use:
  13. Revision History:
  14. --*/
  15. #include <windows.h>
  16. #include <stdio.h>
  17. #include <stdlib.h>
  18. #include "bldrthnk.h"
  19. //
  20. // Internal type definitions follow
  21. //
  22. typedef struct _OBJ {
  23. PCHAR Image;
  24. LONG ImageSize;
  25. PDEFINITIONS Definitions;
  26. } OBJ, *POBJ;
  27. typedef struct _TYPE_REC *PTYPE_REC;
  28. typedef struct _FIELD_REC *PFIELD_REC;
  29. typedef struct _TYPE_REC {
  30. PTYPE_REC Next;
  31. PCHAR Name;
  32. ULONG Size32;
  33. ULONG Size64;
  34. PFIELD_REC FieldList;
  35. BOOL SignExtend;
  36. BOOL Fabricated;
  37. BOOL Only64;
  38. } TYPE_REC;
  39. typedef struct _FIELD_REC {
  40. PFIELD_REC Next;
  41. PCHAR Name;
  42. PCHAR TypeName;
  43. PCHAR SizeFormula;
  44. ULONG TypeSize32;
  45. ULONG TypeSize64;
  46. ULONG Offset32;
  47. ULONG Offset64;
  48. ULONG Size32;
  49. ULONG Size64;
  50. PTYPE_REC TypeRec;
  51. } FIELD_REC;
  52. //
  53. // Static TYPE_REC describing a 64-bit pointer type
  54. //
  55. TYPE_REC pointer64_typerec = {
  56. NULL,
  57. "POINTER64",
  58. 4,
  59. 8,
  60. NULL,
  61. TRUE,
  62. TRUE };
  63. //
  64. // Inline routine to generate a 32-bit pointer value from a ULONGLONG
  65. // value found in an .obj file.
  66. //
  67. __inline
  68. PVOID
  69. CalcPtr(
  70. IN ULONGLONG Address
  71. )
  72. {
  73. return (PVOID)((ULONG)Address);
  74. }
  75. //
  76. // Forward declarations follow
  77. //
  78. VOID
  79. ApplyFixupsToImage(
  80. IN PCHAR ObjImage
  81. );
  82. VOID
  83. __cdecl
  84. CheckCondition(
  85. int Condition,
  86. const char *FormatString,
  87. ...
  88. );
  89. VOID
  90. FabricateMissingTypes(
  91. VOID
  92. );
  93. PTYPE_REC
  94. FindTypeRec(
  95. IN PCHAR Name
  96. );
  97. PVOID
  98. GetMem(
  99. IN ULONG Size
  100. );
  101. VOID
  102. NewGlobalType(
  103. IN PTYPE_REC TypeRec
  104. );
  105. BOOL
  106. ProcessStructure(
  107. IN ULONG StrucIndex
  108. );
  109. void
  110. ReadObj(
  111. IN PCHAR Path,
  112. OUT POBJ Obj
  113. );
  114. int
  115. Usage(
  116. void
  117. );
  118. VOID
  119. WriteCopyList(
  120. IN PTYPE_REC TypeRec
  121. );
  122. VOID
  123. WriteCopyListWorker(
  124. IN PTYPE_REC TypeRec,
  125. IN ULONG Offset32,
  126. IN ULONG Offset64,
  127. IN PCHAR ParentName
  128. );
  129. VOID
  130. WriteCopyRoutine(
  131. IN PTYPE_REC TypeRec
  132. );
  133. VOID
  134. WriteThunkSource(
  135. VOID
  136. );
  137. //
  138. // Global data follows.
  139. //
  140. OBJ Obj32;
  141. OBJ Obj64;
  142. PTYPE_REC TypeRecList = NULL;
  143. int
  144. __cdecl
  145. main(
  146. IN int argc,
  147. IN char *argv[]
  148. )
  149. /*++
  150. Routine Description:
  151. This is the main entrypoint of bldrthnk.exe. Two command-line arguments
  152. are expected: first the path to the 32-bit .obj module, the second to
  153. the path to the 64-bit .obj module. The .objs are expected to be the
  154. result of compiling m4-generated structure definition code.
  155. Arguments:
  156. Return value:
  157. 0 for success, non-zero otherwise.
  158. --*/
  159. {
  160. ULONG strucIndex;
  161. //
  162. // argv[1] is the name of the 32-bit obj, argv[2] is the name of the
  163. // 64-bit obj
  164. //
  165. if (argc != 3) {
  166. return Usage();
  167. }
  168. //
  169. // Usage:
  170. //
  171. // bldrthnk <32-bit.obj> <64-bit.obj>
  172. //
  173. ReadObj( argv[1], &Obj32 );
  174. ReadObj( argv[2], &Obj64 );
  175. //
  176. // Process each STRUC_DEF structure
  177. //
  178. strucIndex = 0;
  179. while (ProcessStructure( strucIndex )) {
  180. strucIndex += 1;
  181. }
  182. FabricateMissingTypes();
  183. //
  184. // Write out the file
  185. //
  186. WriteThunkSource();
  187. return 0;
  188. }
  189. int
  190. Usage(
  191. VOID
  192. )
  193. /*++
  194. Routine Description:
  195. Displays program usage.
  196. Arguments:
  197. Return value:
  198. --*/
  199. {
  200. fprintf(stderr, "Usage: bldrthnk.exe <32-bit obj> <64-bit obj>\n");
  201. return -1;
  202. }
  203. void
  204. ReadObj(
  205. IN PCHAR Path,
  206. OUT POBJ Obj
  207. )
  208. /*++
  209. Routine Description:
  210. Allocates an appropriate buffer, and reads into it the supplied object
  211. image in its entirety.
  212. Arguments:
  213. Path - Supplies the path of the object image to process.
  214. Obj - Supplies a pointer to an OBJ structure which upon return will
  215. updated with the appropriate data.
  216. Return value:
  217. None.
  218. --*/
  219. {
  220. FILE *objFile;
  221. int result;
  222. long objImageSize;
  223. PCHAR objImage;
  224. PULONG sigPtr;
  225. PULONG srchEnd;
  226. PDEFINITIONS definitions;
  227. LONGLONG imageBias;
  228. //
  229. // Open the file
  230. //
  231. objFile = fopen( Path, "rb" );
  232. CheckCondition( objFile != NULL,
  233. "Cannot open %s for reading.\n",
  234. Path );
  235. //
  236. // Get the file size, allocate a buffer, read it in, and close.
  237. //
  238. result = fseek( objFile, 0, SEEK_END );
  239. CheckCondition( result == 0,
  240. "fseek() failed, error %d\n",
  241. errno );
  242. objImageSize = ftell( objFile );
  243. CheckCondition( objImageSize != -1L,
  244. "ftell() failed, error %d\n",
  245. errno );
  246. CheckCondition( objImageSize > 0,
  247. "%s appears to be corrupt\n",
  248. Path );
  249. objImage = GetMem( objImageSize );
  250. result = fseek( objFile, 0, SEEK_SET );
  251. CheckCondition( result == 0,
  252. "fseek() failed, error %d\n",
  253. errno );
  254. result = fread( objImage, 1, objImageSize, objFile );
  255. CheckCondition( result == objImageSize,
  256. "Error reading from %s\n",
  257. Path );
  258. fclose( objFile );
  259. //
  260. // Find the start of the "definitions" array by looking for
  261. // SIG_1 followed by SIG_2
  262. //
  263. srchEnd = (PULONG)(objImage + objImageSize - 2 * sizeof(SIG_1));
  264. sigPtr = (PULONG)objImage;
  265. definitions = NULL;
  266. while (sigPtr < srchEnd) {
  267. if (sigPtr[0] == SIG_1 && sigPtr[1] == SIG_2) {
  268. definitions = (PDEFINITIONS)sigPtr;
  269. break;
  270. }
  271. sigPtr = (PULONG)((PCHAR)sigPtr + 1);
  272. }
  273. CheckCondition( definitions != NULL,
  274. "Error: could not find signature in %s\n",
  275. Path );
  276. //
  277. // Perform fixups on the image
  278. //
  279. ApplyFixupsToImage( objImage );
  280. //
  281. // Fill in the output structure and return
  282. //
  283. Obj->Image = objImage;
  284. Obj->ImageSize = objImageSize;
  285. Obj->Definitions = definitions;
  286. }
  287. VOID
  288. __cdecl
  289. CheckCondition(
  290. int Condition,
  291. const char *FormatString,
  292. ...
  293. )
  294. /*++
  295. Routine Description:
  296. Asserts that Condition is non-zero. If Condition is zero, FormatString
  297. is processed and displayed, and the program is terminated.
  298. Arguments:
  299. Condition - Supplies the boolean value to evaluate.
  300. FormatString, ... - Supplies the format string and optional parameters
  301. to display in the event of a zero Condition.
  302. Return value:
  303. None.
  304. --*/
  305. {
  306. va_list(arglist);
  307. va_start(arglist, FormatString);
  308. if( Condition == 0 ){
  309. //
  310. // A fatal error was encountered. Bail.
  311. //
  312. vprintf( FormatString, arglist );
  313. perror( "genxx" );
  314. exit(-1);
  315. }
  316. }
  317. BOOL
  318. ProcessStructure(
  319. IN ULONG StrucIndex
  320. )
  321. /*++
  322. Routine Description:
  323. Processes a single pair of structure definitions, 32-bit and 64-bit,
  324. respectively.
  325. Processing includes generating a TYPE_REC and associated FIELD_RECs for
  326. the definition pair.
  327. Arguments:
  328. StrucIndex - Supplies the index into the array of STRUC_DEF structures
  329. found within each of the object images.
  330. Return value:
  331. TRUE if the processing was successful, FALSE otherwise (e.g. a terminating
  332. record was located).
  333. --*/
  334. {
  335. PSTRUC_DEF Struc32, Struc64;
  336. PFIELD_DEF Field32, Field64;
  337. ULONG strLen;
  338. ULONG strLen2;
  339. ULONG strLen3;
  340. PTYPE_REC typeRec;
  341. PFIELD_REC fieldRec;
  342. PFIELD_REC insertNode;
  343. BOOL only64;
  344. ULONG index;
  345. Struc32 = CalcPtr( Obj32.Definitions->Structures[ StrucIndex ] );
  346. Struc64 = CalcPtr( Obj64.Definitions->Structures[ StrucIndex ] );
  347. if (Struc64 == NULL) {
  348. return FALSE;
  349. }
  350. if (Struc32 == NULL) {
  351. only64 = TRUE;
  352. } else {
  353. only64 = FALSE;
  354. }
  355. CheckCondition( Struc64 != NULL &&
  356. ((only64 != FALSE) ||
  357. strcmp( Struc32->Name, Struc64->Name ) == 0),
  358. "Mismatched structure definitions found.\n" );
  359. //
  360. // Allocate and build a TYPE_REC for this STRUC_DEF
  361. //
  362. strLen = strlen( Struc64->Name ) + sizeof(char);
  363. typeRec = GetMem( sizeof(TYPE_REC) + strLen );
  364. typeRec->Name = (PCHAR)(typeRec + 1);
  365. typeRec->Only64 = only64;
  366. memcpy( typeRec->Name, Struc64->Name, strLen );
  367. if (only64 == FALSE) {
  368. typeRec->Size32 = Struc32->Size;
  369. }
  370. typeRec->Size64 = Struc64->Size;
  371. typeRec->FieldList = NULL;
  372. typeRec->SignExtend = FALSE;
  373. typeRec->Fabricated = FALSE;
  374. //
  375. // Create the FIELD_RECs hanging off of this type
  376. //
  377. index = 0;
  378. while (TRUE) {
  379. if (only64 == FALSE) {
  380. Field32 = CalcPtr( Struc32->Fields[index] );
  381. }
  382. Field64 = CalcPtr( Struc64->Fields[index] );
  383. if (Field64 == NULL) {
  384. break;
  385. }
  386. if (only64 == FALSE) {
  387. CheckCondition( strcmp( Field32->Name, Field64->Name ) == 0 &&
  388. strcmp( Field32->TypeName, Field64->TypeName ) == 0,
  389. "Mismatched structure definitions found.\n" );
  390. }
  391. strLen = strlen( Field64->Name ) + sizeof(CHAR);
  392. strLen2 = strlen( Field64->TypeName ) + sizeof(CHAR);
  393. strLen3 = strlen( Field64->SizeFormula );
  394. if (strLen3 > 0) {
  395. strLen3 += sizeof(CHAR);
  396. }
  397. fieldRec = GetMem( sizeof(FIELD_REC) + strLen + strLen2 + strLen3 );
  398. fieldRec->Name = (PCHAR)(fieldRec + 1);
  399. fieldRec->TypeName = fieldRec->Name + strLen;
  400. memcpy( fieldRec->Name, Field64->Name, strLen );
  401. memcpy( fieldRec->TypeName, Field64->TypeName, strLen2 );
  402. if (strLen3 > 0) {
  403. fieldRec->SizeFormula = fieldRec->TypeName + strLen2;
  404. memcpy( fieldRec->SizeFormula, Field64->SizeFormula, strLen3 );
  405. } else {
  406. fieldRec->SizeFormula = NULL;
  407. }
  408. if (only64 == FALSE) {
  409. fieldRec->Offset32 = Field32->Offset;
  410. fieldRec->Size32 = Field32->Size;
  411. fieldRec->TypeSize32 = Field32->TypeSize;
  412. }
  413. fieldRec->Offset64 = Field64->Offset;
  414. fieldRec->TypeSize64 = Field64->TypeSize;
  415. fieldRec->Size64 = Field64->Size;
  416. fieldRec->Next = NULL;
  417. fieldRec->TypeRec = NULL;
  418. //
  419. // Insert at the end of the list
  420. //
  421. insertNode = CONTAINING_RECORD( &typeRec->FieldList,
  422. FIELD_REC,
  423. Next );
  424. while (insertNode->Next != NULL) {
  425. insertNode = insertNode->Next;
  426. }
  427. insertNode->Next = fieldRec;
  428. index += 1;
  429. }
  430. //
  431. // Insert it into the global list
  432. //
  433. CheckCondition( FindTypeRec( typeRec->Name ) == NULL,
  434. "Duplicate definition for structure %s\n",
  435. typeRec->Name );
  436. NewGlobalType( typeRec );
  437. return TRUE;
  438. }
  439. PTYPE_REC
  440. FindTypeRec(
  441. IN PCHAR Name
  442. )
  443. /*++
  444. Routine Description:
  445. Searches the global list of TYPE_REC structures for one with a name
  446. that matches the supplied name.
  447. Arguments:
  448. Name - pointer to a null-terminated string representing the name of
  449. the sought type.
  450. Return value:
  451. A pointer to the matching TYPE_REC, or NULL if a match was not found.
  452. --*/
  453. {
  454. PTYPE_REC typeRec;
  455. typeRec = TypeRecList;
  456. while (typeRec != NULL) {
  457. if (strcmp( Name, typeRec->Name ) == 0) {
  458. return typeRec;
  459. }
  460. typeRec = typeRec->Next;
  461. }
  462. return NULL;
  463. }
  464. PVOID
  465. GetMem(
  466. IN ULONG Size
  467. )
  468. /*++
  469. Routine Description:
  470. Memory allocator. Works just like malloc() except that triggers a
  471. fatal error in the event of an out-of-memory condition.
  472. Arguments:
  473. Size - number of bytes to allocate.
  474. Return value:
  475. Returns a pointer to a block of memory of the specified size.
  476. --*/
  477. {
  478. PVOID mem;
  479. mem = malloc( Size );
  480. CheckCondition( mem != NULL,
  481. "Out of memory.\n" );
  482. return mem;
  483. }
  484. VOID
  485. FabricateMissingTypes(
  486. VOID
  487. )
  488. /*++
  489. Routine Description:
  490. Routine to generate TYPE_REC records for simple types referenced, but
  491. not defined, by a structure layout file.
  492. Arguments:
  493. None.
  494. Return value:
  495. None.
  496. --*/
  497. {
  498. PTYPE_REC typeRec;
  499. PTYPE_REC fieldTypeRec;
  500. PFIELD_REC fieldRec;
  501. PCHAR fieldTypeName;
  502. ULONG strLen;
  503. typeRec = TypeRecList;
  504. while (typeRec != NULL) {
  505. fieldRec = typeRec->FieldList;
  506. while (fieldRec != NULL) {
  507. fieldTypeRec = FindTypeRec( fieldRec->TypeName );
  508. if (fieldTypeRec == NULL) {
  509. if (typeRec->Only64 == FALSE) {
  510. CheckCondition( (fieldRec->Size32 == fieldRec->Size64) ||
  511. ((fieldRec->Size32 == 1 ||
  512. fieldRec->Size32 == 2 ||
  513. fieldRec->Size32 == 4 ||
  514. fieldRec->Size32 == 8) &&
  515. (fieldRec->Size64 > fieldRec->Size32) &&
  516. (fieldRec->Size64 % fieldRec->Size32 == 0)),
  517. "Must specify type %s (%s)\n",
  518. fieldRec->TypeName,
  519. typeRec->Name );
  520. }
  521. //
  522. // No typerec exists for this type. Assume it is a simple
  523. // type.
  524. //
  525. if ((typeRec->Only64 != FALSE &&
  526. fieldRec->Size64 == sizeof(ULONGLONG) &&
  527. *fieldRec->TypeName == 'P') ||
  528. (fieldRec->Size32 == sizeof(PVOID) &&
  529. fieldRec->Size64 == sizeof(ULONGLONG))) {
  530. //
  531. // Either a pointer or [U]LONG_PTR type. Make
  532. // it longlong.
  533. //
  534. fieldTypeRec = &pointer64_typerec;
  535. } else {
  536. //
  537. // Some other type.
  538. //
  539. strLen = strlen( fieldRec->TypeName ) + sizeof(CHAR);
  540. fieldTypeRec = GetMem( sizeof(TYPE_REC) + strLen );
  541. fieldTypeRec->Name = (PCHAR)(fieldTypeRec + 1);
  542. memcpy( fieldTypeRec->Name, fieldRec->TypeName, strLen );
  543. fieldTypeRec->Size32 = fieldRec->Size32;
  544. fieldTypeRec->Size64 = fieldRec->Size64;
  545. fieldTypeRec->FieldList = NULL;
  546. fieldTypeRec->SignExtend = TRUE;
  547. fieldTypeRec->Fabricated = TRUE;
  548. NewGlobalType( fieldTypeRec );
  549. }
  550. }
  551. fieldRec->TypeRec = fieldTypeRec;
  552. fieldRec = fieldRec->Next;
  553. }
  554. typeRec = typeRec->Next;
  555. }
  556. }
  557. VOID
  558. WriteCopyRecord(
  559. IN ULONG Offset32,
  560. IN ULONG Offset64,
  561. IN PCHAR TypeName,
  562. IN ULONG Size32,
  563. IN ULONG Size64,
  564. IN BOOL SignExtend,
  565. IN PCHAR FieldName,
  566. IN BOOL Last
  567. )
  568. /*++
  569. Routine Description:
  570. Support routine to generate the text of a copy record.
  571. Arguments:
  572. Offset32 - Offset of this field within a 32-bit structure layout
  573. Offset64 - Offset of this field within a 64-bit structure layout
  574. Size32 - Size of this field within a 32-bit structure layout
  575. Size64 - Size of this field within a 64-bit structure layout
  576. SignExtend - Indicates whether this type should be sign extended or not
  577. FieldName - Name of the field
  578. Last - Whether this is the last copy record in a zero-terminated list
  579. Return value:
  580. None
  581. --*/
  582. {
  583. CHAR buf[ 255 ];
  584. if (SignExtend) {
  585. sprintf(buf,"IS_SIGNED_TYPE(%s)", TypeName);
  586. } else {
  587. sprintf(buf,"FALSE");
  588. }
  589. printf(" { \t0x%x, \t0x%x, \t0x%x, \t0x%x, \t%5s }%s\n",
  590. Offset32,
  591. Offset64,
  592. Size32,
  593. Size64,
  594. buf,
  595. Last ? "" : "," );
  596. }
  597. VOID
  598. WriteDefinition64(
  599. IN PTYPE_REC TypeRec
  600. )
  601. /*++
  602. Routine Description:
  603. Generates a structure definition that represents, to a 32-bit compiler,
  604. the layout of a 64-bit structure.
  605. Arguments:
  606. TypeRec - Pointer to the TYPE_REC structure defining this type.
  607. Return value:
  608. None.
  609. --*/
  610. {
  611. PFIELD_REC fieldRec;
  612. ULONG currentOffset;
  613. PTYPE_REC fieldTypeRec;
  614. ULONG padBytes;
  615. ULONG reservedCount;
  616. currentOffset = 0;
  617. reservedCount = 0;
  618. printf("typedef struct _%s_64 {\n", TypeRec->Name );
  619. fieldRec = TypeRec->FieldList;
  620. while (fieldRec != NULL) {
  621. fieldTypeRec = fieldRec->TypeRec;
  622. padBytes = fieldRec->Offset64 - currentOffset;
  623. if (padBytes > 0) {
  624. printf(" UCHAR Reserved%d[ 0x%x ];\n",
  625. reservedCount,
  626. padBytes );
  627. currentOffset += padBytes;
  628. reservedCount += 1;
  629. }
  630. printf(" %s%s %s",
  631. fieldTypeRec->Name,
  632. fieldTypeRec->Fabricated ? "" : "_64",
  633. fieldRec->Name );
  634. if (fieldRec->Size64 > fieldRec->TypeSize64) {
  635. CheckCondition( fieldRec->Size64 % fieldRec->TypeSize64 == 0,
  636. "Internal error type %s.%s\n",
  637. TypeRec->Name, fieldRec->Name );
  638. //
  639. // This field must be an array
  640. //
  641. printf("[%d]", fieldRec->Size64 / fieldRec->TypeSize64);
  642. }
  643. printf(";\n");
  644. currentOffset += fieldRec->Size64;
  645. fieldRec = fieldRec->Next;
  646. }
  647. padBytes = TypeRec->Size64 - currentOffset;
  648. if (padBytes > 0) {
  649. printf(" UCHAR Reserved%d[ 0x%x ];\n", reservedCount, padBytes );
  650. currentOffset += padBytes;
  651. reservedCount += 1;
  652. }
  653. printf("} %s_64, *P%s_64;\n\n", TypeRec->Name, TypeRec->Name );
  654. fieldRec = TypeRec->FieldList;
  655. while (fieldRec != NULL) {
  656. fieldTypeRec = fieldRec->TypeRec;
  657. printf("C_ASSERT( FIELD_OFFSET(%s_64,%s) == 0x%x);\n",
  658. TypeRec->Name,
  659. fieldRec->Name,
  660. fieldRec->Offset64);
  661. fieldRec = fieldRec->Next;
  662. }
  663. printf("\n");
  664. }
  665. VOID
  666. WriteCopyList(
  667. IN PTYPE_REC TypeRec
  668. )
  669. /*++
  670. Routine Description:
  671. Generates the list of copy records necessary to copy the contents of
  672. each of the fields with TypeRec from their 32-bit layout to their
  673. 64-bit layout.
  674. Arguments:
  675. TypeRec - Pointer to the TYPE_REC structure that defines this type.
  676. Return value:
  677. None.
  678. --*/
  679. {
  680. PFIELD_REC fieldRec;
  681. printf("COPY_REC cr3264_%s[] = {\n", TypeRec->Name);
  682. WriteCopyListWorker( TypeRec, 0, 0, NULL );
  683. WriteCopyRecord( 0,0,NULL,0,0,FALSE,NULL,TRUE );
  684. printf("};\n\n");
  685. }
  686. VOID
  687. WriteCopyListWorker(
  688. IN PTYPE_REC TypeRec,
  689. IN ULONG Offset32,
  690. IN ULONG Offset64,
  691. IN PCHAR ParentName
  692. )
  693. /*++
  694. Routine Description:
  695. Recursively-called support routine for WriteCopyList. This routine
  696. generates a copy record if this type is not composed of child types,
  697. otherwise it calls itself recursively for each child type.
  698. Arguments:
  699. TypeRec - Pointer to the definition of the type to process.
  700. Offset32 - Current offset within the master structure being defined.
  701. Offset64 - Current offset within the master structure being defined.
  702. ParentName - Not currently used.
  703. Return value:
  704. None.
  705. --*/
  706. {
  707. PFIELD_REC fieldRec;
  708. PTYPE_REC typeRec;
  709. CHAR fieldName[ 255 ];
  710. PCHAR fieldStart;
  711. fieldRec = TypeRec->FieldList;
  712. if (fieldRec == NULL) {
  713. WriteCopyRecord( Offset32,
  714. Offset64,
  715. TypeRec->Name,
  716. TypeRec->Size32,
  717. TypeRec->Size64,
  718. TypeRec->SignExtend,
  719. ParentName,
  720. FALSE );
  721. } else {
  722. //
  723. // Build the field name
  724. //
  725. if (ParentName != NULL) {
  726. strcpy( fieldName, ParentName );
  727. strcat( fieldName, "." );
  728. } else {
  729. fieldName[0] = '\0';
  730. }
  731. fieldStart = &fieldName[ strlen(fieldName) ];
  732. do {
  733. strcpy( fieldStart, fieldRec->Name );
  734. // typeRec = FindTypeRec( fieldRec->TypeName );
  735. typeRec = fieldRec->TypeRec;
  736. WriteCopyListWorker( typeRec,
  737. fieldRec->Offset32 + Offset32,
  738. fieldRec->Offset64 + Offset64,
  739. fieldName );
  740. fieldRec = fieldRec->Next;
  741. } while (fieldRec != NULL);
  742. }
  743. }
  744. VOID
  745. WriteBufferCopies(
  746. IN PTYPE_REC TypeRec,
  747. IN PCHAR StrucName
  748. )
  749. {
  750. PFIELD_REC fieldRec;
  751. PTYPE_REC typeRec;
  752. CHAR strucName[ 255 ];
  753. CHAR sizeFormula[ 255 ];
  754. PCHAR fieldPos;
  755. PCHAR src;
  756. PCHAR dst;
  757. if (TypeRec == NULL) {
  758. return;
  759. }
  760. strcpy(strucName,StrucName );
  761. if (*StrucName != '\0') {
  762. strcat(strucName,".");
  763. }
  764. fieldPos = &strucName[ strlen(strucName) ];
  765. fieldRec = TypeRec->FieldList;
  766. while (fieldRec != NULL) {
  767. strcpy(fieldPos,fieldRec->Name);
  768. if (fieldRec->SizeFormula != NULL) {
  769. //
  770. // Perform substitution on the size formula
  771. //
  772. dst = sizeFormula;
  773. src = fieldRec->SizeFormula;
  774. do {
  775. if (*src == '%' &&
  776. *(src+1) == '1') {
  777. dst += sprintf(dst,
  778. "Source->%s%s",
  779. StrucName,
  780. *StrucName == '\0' ? "" : ".");
  781. src += 2;
  782. } else {
  783. *dst++ = *src++;
  784. }
  785. } while (*src != '\0');
  786. *dst = '\0';
  787. printf("\n"
  788. " status = \n"
  789. " CopyBuf( Source->%s,\n"
  790. " &Destination->%s,\n"
  791. " %s );\n"
  792. " if (status != ESUCCESS) {\n"
  793. " return status;\n"
  794. " }\n",
  795. strucName,
  796. strucName,
  797. sizeFormula);
  798. }
  799. typeRec = fieldRec->TypeRec;
  800. WriteBufferCopies( typeRec, strucName );
  801. fieldRec = fieldRec->Next;
  802. }
  803. }
  804. VOID
  805. WriteThunkSource(
  806. VOID
  807. )
  808. /*++
  809. Routine Description:
  810. Generates the source code and supporting definitions necessary to
  811. copy all or portions of the contents of 32-bit structures to the
  812. equivalent 64-bit layout.
  813. Arguments:
  814. None.
  815. Return value:
  816. None.
  817. --*/
  818. {
  819. PTYPE_REC typeRec;
  820. printf("//\n");
  821. printf("// Autogenerated file, do not edit\n");
  822. printf("//\n\n");
  823. printf("#include <bldrthnk.h>\n\n");
  824. printf("#pragma warning(disable:4296)\n\n");
  825. //
  826. // Output the 64-bit type definitions
  827. //
  828. printf("#pragma pack(push,1)\n\n");
  829. typeRec = TypeRecList;
  830. while (typeRec != NULL) {
  831. if (typeRec->Fabricated == FALSE) {
  832. WriteDefinition64( typeRec );
  833. }
  834. typeRec = typeRec->Next;
  835. }
  836. printf("#pragma pack(pop)\n\n");
  837. //
  838. // Output the copy records
  839. //
  840. typeRec = TypeRecList;
  841. while (typeRec != NULL) {
  842. if (typeRec->Only64 == FALSE &&
  843. typeRec->Fabricated == FALSE) {
  844. WriteCopyList( typeRec );
  845. }
  846. typeRec = typeRec->Next;
  847. }
  848. //
  849. // Generate the copy routines
  850. //
  851. typeRec = TypeRecList;
  852. while (typeRec != NULL) {
  853. if (typeRec->Only64 == FALSE &&
  854. typeRec->Fabricated == FALSE) {
  855. WriteCopyRoutine( typeRec );
  856. }
  857. typeRec = typeRec->Next;
  858. }
  859. printf("\n");
  860. }
  861. VOID
  862. WriteCopyRoutine(
  863. IN PTYPE_REC TypeRec
  864. )
  865. /*++
  866. Routine Description:
  867. Generates text that implements a function to copy the contents of a
  868. structure of the specified type from a 32-bit layout to a 64-bit
  869. layout.
  870. Arguments:
  871. TypeRec - Pointer to the type for which the function should be generated.
  872. Return value:
  873. None.
  874. --*/
  875. {
  876. PCHAR typeName;
  877. typeName = TypeRec->Name;
  878. printf("\n"
  879. "ARC_STATUS\n"
  880. "__inline\n"
  881. "static\n"
  882. "Copy_%s(\n"
  883. " IN %s *Source,\n"
  884. " OUT %s_64 *Destination\n"
  885. " )\n"
  886. "{\n"
  887. " ARC_STATUS status = ESUCCESS;"
  888. "\n"
  889. " DbgPrint(\"BLAMD64: Copy %s->%s_64 (0x%%08x->0x%%08x)\\n\",\n"
  890. " (ULONG)Source, (ULONG)Destination );\n"
  891. "\n"
  892. " CopyRec( Source, Destination, cr3264_%s );\n",
  893. typeName,
  894. typeName,
  895. typeName,
  896. typeName,
  897. typeName,
  898. typeName );
  899. WriteBufferCopies( TypeRec, "" );
  900. printf(" return status;\n");
  901. printf("}\n\n");
  902. }
  903. VOID
  904. ApplyFixupsToImage(
  905. IN PCHAR ObjImage
  906. )
  907. /*++
  908. Routine Description:
  909. Processes fixup records found within an object image.
  910. Arguments:
  911. Pointer to a buffer containing the entire image.
  912. Return value:
  913. None.
  914. --*/
  915. {
  916. //
  917. // Applies fixups to the OBJ image loaded at ObjImage
  918. //
  919. PIMAGE_FILE_HEADER fileHeader;
  920. PIMAGE_SECTION_HEADER sectionHeader;
  921. PIMAGE_SECTION_HEADER sectionHeaderArray;
  922. PIMAGE_SYMBOL symbolTable;
  923. PIMAGE_SYMBOL symbol;
  924. PIMAGE_RELOCATION reloc;
  925. PIMAGE_RELOCATION relocArray;
  926. ULONG sectionNum;
  927. ULONG relocNum;
  928. ULONG_PTR targetVa;
  929. PULONG_PTR fixupVa;
  930. fileHeader = (PIMAGE_FILE_HEADER)ObjImage;
  931. //
  932. // We need the symbol table to apply the fixups
  933. //
  934. symbolTable = (PIMAGE_SYMBOL)(ObjImage +
  935. fileHeader->PointerToSymbolTable);
  936. //
  937. // Get a pointer to the first element in the section header
  938. //
  939. sectionHeaderArray = (PIMAGE_SECTION_HEADER)(ObjImage +
  940. sizeof( IMAGE_FILE_HEADER ) +
  941. fileHeader->SizeOfOptionalHeader);
  942. //
  943. // Apply the fixups for each section
  944. //
  945. for( sectionNum = 0;
  946. sectionNum < fileHeader->NumberOfSections;
  947. sectionNum++ ){
  948. sectionHeader = &sectionHeaderArray[ sectionNum ];
  949. //
  950. // Apply each fixup in this section
  951. //
  952. relocArray = (PIMAGE_RELOCATION)(ObjImage +
  953. sectionHeader->PointerToRelocations);
  954. for( relocNum = 0;
  955. relocNum < sectionHeader->NumberOfRelocations;
  956. relocNum++ ){
  957. reloc = &relocArray[ relocNum ];
  958. //
  959. // The relocation gives us the position in the image of the
  960. // relocation modification (VirtualAddress). To find out what
  961. // to put there, we have to look the symbol up in the symbol index.
  962. //
  963. symbol = &symbolTable[ reloc->SymbolTableIndex ];
  964. targetVa =
  965. sectionHeaderArray[ symbol->SectionNumber-1 ].PointerToRawData;
  966. targetVa += symbol->Value;
  967. targetVa += (ULONG_PTR)ObjImage;
  968. fixupVa = (PULONG_PTR)(ObjImage +
  969. reloc->VirtualAddress +
  970. sectionHeader->PointerToRawData );
  971. *fixupVa = targetVa;
  972. }
  973. }
  974. }
  975. VOID
  976. NewGlobalType(
  977. IN PTYPE_REC TypeRec
  978. )
  979. /*++
  980. Routine Description:
  981. Inserts a new TYPE_REC structure at the end of the global TYPE_REC
  982. list.
  983. Arguments:
  984. TypeRec - Pointer to the TYPE_REC structure to insert.
  985. Return value:
  986. None.
  987. --*/
  988. {
  989. PTYPE_REC insertNode;
  990. insertNode = CONTAINING_RECORD( &TypeRecList,
  991. TYPE_REC,
  992. Next );
  993. while (insertNode->Next != NULL) {
  994. insertNode = insertNode->Next;
  995. }
  996. insertNode->Next = TypeRec;
  997. TypeRec->Next = NULL;
  998. }