pefile.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. /* This file is part of DiDiExtract.
  2. *
  3. * DiDiExtract is free software: you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License as published by
  5. * the Free Software Foundation, either version 3 of the License, or
  6. * (at your option) any later version.
  7. *
  8. * DiDiExtract 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. *
  13. * You should have received a copy of the GNU General Public License
  14. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  15. */
  16. #include <stdio.h>
  17. #include <errno.h>
  18. #include "reader.h"
  19. #include "pefile.h"
  20. #include "print.h"
  21. /* pefileGetOverlayOffset() - Get the overlay-offset of a PE file.
  22. *
  23. * @returns: The offset to overlay-data on success, <0 on error. */
  24. long pefileGetOverlayOffset(const char * filePath) {
  25. long fileSize;
  26. FILE * fp;
  27. size_t noRead;
  28. long overlayOffset;
  29. // Open file
  30. fp = fopen(filePath, "rb");
  31. // Failed to open file
  32. if (fp == NULL) {
  33. printError("pefileGetOverlayOffset failed to open '%s'\n", filePath);
  34. return -1;
  35. }
  36. // Determine file size
  37. if (fseek(fp, 0l, SEEK_END) != 0) {
  38. printError("pefileGetOverlayOffset failed to seek to end of file '%s'\n",
  39. filePath);
  40. fclose(fp);
  41. return -1;
  42. }
  43. if ((fileSize = ftell(fp)) < 1) {
  44. printError("pefileGetOverlayOffset failed to determine file size of file "
  45. "'%s'\n", filePath);
  46. fclose(fp);
  47. return -1;
  48. }
  49. // Set cursor back to start of the file
  50. if (fseek(fp, 0l, SEEK_SET) != 0) {
  51. printError("pefileGetOverlayOffset failed to seek to start of file '%s'\n",
  52. filePath);
  53. fclose(fp);
  54. return -1;
  55. }
  56. // Read MsDosHeader
  57. size_t readSize = sizeof(MsDosHeader);
  58. if (readSize > (size_t)fileSize) {
  59. printError("pefileGetOverlayOffset file is to small to contain a MS Dos "
  60. "header. File: '%s'\n", filePath);
  61. fclose(fp);
  62. return -1;
  63. }
  64. MsDosHeader msDosHeader;
  65. noRead = fread(&msDosHeader, readSize, 1, fp);
  66. if (noRead != 1) {
  67. printError("pefileGetOverlayOffset failed to read the MS Dos header. "
  68. "noRead: %ld readSize: %ld\n", noRead, readSize);
  69. fclose(fp);
  70. return -1;
  71. }
  72. if (msDosHeader.signature != 0x5A4D) {
  73. printError("pefileGetOverlayOffset this is not a PE file for sure. The "
  74. "MS-DOS header signature doesn't match (not MZ).\n");
  75. fclose(fp);
  76. return -1;
  77. }
  78. // Read e_lfanew (offset to PE file header)
  79. uint32_t e_lfanew;
  80. if (fseek(fp, 0x3C, SEEK_SET) != 0) {
  81. printError("pefileGetOverlayOffset failed to seek to 0x3C.\n");
  82. printError("pefileGetOverlayOffset errno: %s\n", strerror(errno));
  83. fclose(fp);
  84. return -1;
  85. }
  86. if (fread(&e_lfanew, 4, 1, fp) != 1) {
  87. printError("pefileGetOverlayOffset failed to read e_lfanew.\n");
  88. fclose(fp);
  89. return -1;
  90. }
  91. if (e_lfanew >= fileSize) {
  92. printError("pefileGetOverlayOffset PE file offset is larger then file "
  93. "size.\n");
  94. fclose(fp);
  95. return -1;
  96. }
  97. // Read PE File Header
  98. if (fseek(fp, (long)e_lfanew, SEEK_SET) != 0) {
  99. printError("pefileGetOverlayOffset failed to seek to e_lfanew.\n");
  100. printError("pefileGetOverlayOffset errno: %s\n", strerror(errno));
  101. fclose(fp);
  102. return -1;
  103. }
  104. PeFileHeader peFileHeader;
  105. if (fread(&peFileHeader, sizeof(PeFileHeader), 1, fp) != 1) {
  106. printError("pefileGetOverlayOffset failed to read PE File Header\n");
  107. fclose(fp);
  108. return -1;
  109. }
  110. if ((peFileHeader.signature << 16) != 0x45500000) {
  111. printError("pefileGetOverlayOffset this is not a PE file for sure (2).\n");
  112. fclose(fp);
  113. return -1;
  114. }
  115. // Skip optional header
  116. if (peFileHeader.optionalHeaderSize > 0) {
  117. if (fseek(fp, peFileHeader.optionalHeaderSize, SEEK_CUR) != 0) {
  118. printError("pefileGetOverlayOffset failed to skip over the optional "
  119. "header.\n");
  120. printError("pefileGetOverlayOffset errno: %s\n", strerror(errno));
  121. fclose(fp);
  122. return -1;
  123. }
  124. }
  125. // Read sections
  126. overlayOffset = 0l;
  127. for (uint32_t i=0; i< peFileHeader.numberOfSections; i++) {
  128. PeImageSectionHeader sectionHeader;
  129. if (fread(&sectionHeader, sizeof(PeImageSectionHeader), 1, fp) != 1) {
  130. printError("pefileGetOverlayOffset failed to read section header.\n");
  131. fclose(fp);
  132. return -1;
  133. }
  134. if ((sectionHeader.rawDataLocation + sectionHeader.rawDataSize) > overlayOffset) {
  135. overlayOffset = sectionHeader.rawDataLocation + sectionHeader.rawDataSize;
  136. }
  137. }
  138. fclose(fp);
  139. if (overlayOffset > fileSize) {
  140. printError("pefileGetOverlayOffset no overlay offset larger then file.\n");
  141. return -1;
  142. }
  143. return overlayOffset;
  144. }