mirror of
https://github.com/QuasarApp/pe-parse.git
synced 2025-04-27 12:54:31 +00:00
When read 2, 4, or 8 bytes from a bounded_buffer, we only checked to see if the offset, not the whole span, was in bounds. This results in an arbitrary memory read of up to 1, 3, or 7 bytes when the offset is aligned with the very end of the buffer.
222 lines
6.7 KiB
C++
222 lines
6.7 KiB
C++
/*
|
|
The MIT License (MIT)
|
|
|
|
Copyright (c) 2013 Andrew Ruef
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
THE SOFTWARE.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <cstdint>
|
|
#include <string>
|
|
|
|
#include "nt-headers.h"
|
|
#include "to_string.h"
|
|
|
|
#ifdef _MSC_VER
|
|
#define __typeof__(x) std::remove_reference < decltype(x) > ::type
|
|
#endif
|
|
|
|
#define PE_ERR(x) \
|
|
err = static_cast<pe_err>(x); \
|
|
err_loc.assign(__func__); \
|
|
err_loc += ":" + to_string<std::uint32_t>(__LINE__, std::dec);
|
|
|
|
#define READ_WORD(b, o, inst, member) \
|
|
if (!readWord(b, o + _offset(__typeof__(inst), member), inst.member)) { \
|
|
PE_ERR(PEERR_READ); \
|
|
return false; \
|
|
}
|
|
|
|
#define READ_DWORD(b, o, inst, member) \
|
|
if (!readDword(b, o + _offset(__typeof__(inst), member), inst.member)) { \
|
|
PE_ERR(PEERR_READ); \
|
|
return false; \
|
|
}
|
|
|
|
#define READ_QWORD(b, o, inst, member) \
|
|
if (!readQword(b, o + _offset(__typeof__(inst), member), inst.member)) { \
|
|
PE_ERR(PEERR_READ); \
|
|
return false; \
|
|
}
|
|
|
|
#define READ_BYTE(b, o, inst, member) \
|
|
if (!readByte(b, o + _offset(__typeof__(inst), member), inst.member)) { \
|
|
PE_ERR(PEERR_READ); \
|
|
return false; \
|
|
}
|
|
|
|
#define TEST_MACHINE_CHARACTERISTICS(h, m, ch) \
|
|
((h.FileHeader.Machine == m) && (h.FileHeader.Characteristics & ch))
|
|
|
|
namespace peparse {
|
|
|
|
typedef std::uint32_t RVA;
|
|
typedef std::uint64_t VA;
|
|
|
|
struct buffer_detail;
|
|
|
|
typedef struct _bounded_buffer {
|
|
std::uint8_t *buf;
|
|
std::uint32_t bufLen;
|
|
bool copy;
|
|
bool swapBytes;
|
|
buffer_detail *detail;
|
|
} bounded_buffer;
|
|
|
|
struct resource {
|
|
resource()
|
|
: type(0), name(0), lang(0), codepage(0), RVA(0), size(0), buf(nullptr) {
|
|
}
|
|
|
|
std::string type_str;
|
|
std::string name_str;
|
|
std::string lang_str;
|
|
std::uint32_t type;
|
|
std::uint32_t name;
|
|
std::uint32_t lang;
|
|
std::uint32_t codepage;
|
|
std::uint32_t RVA;
|
|
std::uint32_t size;
|
|
bounded_buffer *buf;
|
|
};
|
|
|
|
// http://msdn.microsoft.com/en-us/library/ms648009(v=vs.85).aspx
|
|
enum resource_type {
|
|
RT_CURSOR = 1,
|
|
RT_BITMAP = 2,
|
|
RT_ICON = 3,
|
|
RT_MENU = 4,
|
|
RT_DIALOG = 5,
|
|
RT_STRING = 6,
|
|
RT_FONTDIR = 7,
|
|
RT_FONT = 8,
|
|
RT_ACCELERATOR = 9,
|
|
RT_RCDATA = 10,
|
|
RT_MESSAGETABLE = 11,
|
|
RT_GROUP_CURSOR = 12, // MAKEINTRESOURCE((ULONG_PTR)(RT_CURSOR) + 11)
|
|
RT_GROUP_ICON = 14, // MAKEINTRESOURCE((ULONG_PTR)(RT_ICON) + 11)
|
|
RT_VERSION = 16,
|
|
RT_DLGINCLUDE = 17,
|
|
RT_PLUGPLAY = 19,
|
|
RT_VXD = 20,
|
|
RT_ANICURSOR = 21,
|
|
RT_ANIICON = 22,
|
|
RT_HTML = 23,
|
|
RT_MANIFEST = 24
|
|
};
|
|
|
|
enum pe_err {
|
|
PEERR_NONE = 0,
|
|
PEERR_MEM = 1,
|
|
PEERR_HDR = 2,
|
|
PEERR_SECT = 3,
|
|
PEERR_RESC = 4,
|
|
PEERR_SECTVA = 5,
|
|
PEERR_READ = 6,
|
|
PEERR_OPEN = 7,
|
|
PEERR_STAT = 8,
|
|
PEERR_MAGIC = 9,
|
|
PEERR_BUFFER = 10,
|
|
PEERR_ADDRESS = 11,
|
|
};
|
|
|
|
bool readByte(bounded_buffer *b, std::uint32_t offset, std::uint8_t &out);
|
|
bool readWord(bounded_buffer *b, std::uint32_t offset, std::uint16_t &out);
|
|
bool readDword(bounded_buffer *b, std::uint32_t offset, std::uint32_t &out);
|
|
bool readQword(bounded_buffer *b, std::uint32_t offset, std::uint64_t &out);
|
|
|
|
bounded_buffer *readFileToFileBuffer(const char *filePath);
|
|
bounded_buffer *
|
|
splitBuffer(bounded_buffer *b, std::uint32_t from, std::uint32_t to);
|
|
void deleteBuffer(bounded_buffer *b);
|
|
uint64_t bufLen(bounded_buffer *b);
|
|
|
|
struct parsed_pe_internal;
|
|
|
|
typedef struct _pe_header {
|
|
nt_header_32 nt;
|
|
} pe_header;
|
|
|
|
typedef struct _parsed_pe {
|
|
bounded_buffer *fileBuffer;
|
|
parsed_pe_internal *internal;
|
|
pe_header peHeader;
|
|
} parsed_pe;
|
|
|
|
// get parser error status as integer
|
|
std::uint32_t GetPEErr();
|
|
|
|
// get parser error status as string
|
|
std::string GetPEErrString();
|
|
|
|
// get parser error location as string
|
|
std::string GetPEErrLoc();
|
|
|
|
// get a PE parse context from a file
|
|
parsed_pe *ParsePEFromFile(const char *filePath);
|
|
|
|
// destruct a PE context
|
|
void DestructParsedPE(parsed_pe *p);
|
|
|
|
// iterate over the resources
|
|
typedef int (*iterRsrc)(void *, resource);
|
|
void IterRsrc(parsed_pe *pe, iterRsrc cb, void *cbd);
|
|
|
|
// iterate over the imports by RVA and string
|
|
typedef int (*iterVAStr)(void *, VA, std::string &, std::string &);
|
|
void IterImpVAString(parsed_pe *pe, iterVAStr cb, void *cbd);
|
|
|
|
// iterate over relocations in the PE file
|
|
typedef int (*iterReloc)(void *, VA, reloc_type);
|
|
void IterRelocs(parsed_pe *pe, iterReloc cb, void *cbd);
|
|
|
|
// Iterate over symbols (symbol table) in the PE file
|
|
typedef int (*iterSymbol)(void *,
|
|
std::string &,
|
|
std::uint32_t &,
|
|
std::int16_t &,
|
|
std::uint16_t &,
|
|
std::uint8_t &,
|
|
std::uint8_t &);
|
|
void IterSymbols(parsed_pe *pe, iterSymbol cb, void *cbd);
|
|
|
|
// iterate over the exports
|
|
typedef int (*iterExp)(void *, VA, std::string &, std::string &);
|
|
void IterExpVA(parsed_pe *pe, iterExp cb, void *cbd);
|
|
|
|
// iterate over sections
|
|
typedef int (*iterSec)(
|
|
void *, VA secBase, std::string &, image_section_header, bounded_buffer *b);
|
|
void IterSec(parsed_pe *pe, iterSec cb, void *cbd);
|
|
|
|
// get byte at VA in PE
|
|
bool ReadByteAtVA(parsed_pe *pe, VA v, std::uint8_t &b);
|
|
|
|
// get entry point into PE
|
|
bool GetEntryPoint(parsed_pe *pe, VA &v);
|
|
|
|
// get machine as human readable string
|
|
const char *GetMachineAsString(parsed_pe *pe);
|
|
|
|
// get subsystem as human readable string
|
|
const char *GetSubsystemAsString(parsed_pe *pe);
|
|
} // namespace peparse
|