mirror of
https://github.com/mtrojnar/osslsigncode.git
synced 2025-04-05 09:08:04 -05:00
Check PE header value ranges, CID 1519382
This commit is contained in:
parent
5981c740c9
commit
5fef4faf47
@ -1562,27 +1562,62 @@ static u_char *pe_calc_page_hash(char *indata, uint32_t header_size,
|
|||||||
uint32_t pe32plus, uint32_t sigpos, int phtype, int *rphlen)
|
uint32_t pe32plus, uint32_t sigpos, int phtype, int *rphlen)
|
||||||
{
|
{
|
||||||
uint16_t nsections, opthdr_size;
|
uint16_t nsections, opthdr_size;
|
||||||
uint32_t pagesize, hdrsize;
|
uint32_t alignment, pagesize, hdrsize;
|
||||||
uint32_t rs, ro, l, lastpos = 0;
|
uint32_t rs, ro, l, lastpos = 0;
|
||||||
int pphlen, phlen, i, pi = 1;
|
int pphlen, phlen, i, pi = 1;
|
||||||
u_char *res, *zeroes;
|
u_char *res, *zeroes;
|
||||||
char *sections;
|
char *sections;
|
||||||
const EVP_MD *md = EVP_get_digestbynid(phtype);
|
const EVP_MD *md = EVP_get_digestbynid(phtype);
|
||||||
EVP_MD_CTX *mdctx = EVP_MD_CTX_new();
|
EVP_MD_CTX *mdctx;
|
||||||
|
|
||||||
|
/* NumberOfSections indicates the size of the section table,
|
||||||
|
* which immediately follows the headers. */
|
||||||
|
nsections = GET_UINT16_LE(indata + header_size + 6);
|
||||||
|
if (nsections == 0 || nsections > UINT16_MAX) {
|
||||||
|
printf("Corrupted number of sections: 0x%08X\n", nsections);
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
}
|
||||||
|
/* FileAlignment is the alignment factor (in bytes) that is used to align
|
||||||
|
* the raw data of sections in the image file. The value should be a power
|
||||||
|
* of 2 between 512 and 64 K, inclusive. The default is 512. */
|
||||||
|
alignment = GET_UINT32_LE(indata + header_size + 60);
|
||||||
|
if (alignment < 512 || alignment > UINT16_MAX) {
|
||||||
|
printf("Corrupted file alignment factor: 0x%08X\n", alignment);
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
}
|
||||||
|
/* SectionAlignment is the alignment (in bytes) of sections when they are
|
||||||
|
* loaded into memory. It must be greater than or equal to FileAlignment.
|
||||||
|
* The default is the page size for the architecture. */
|
||||||
|
pagesize = GET_UINT32_LE(indata + header_size + 56);
|
||||||
|
if (pagesize < alignment || pagesize > UINT32_MAX) {
|
||||||
|
printf("Corrupted page size: 0x%08X\n", pagesize);
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
}
|
||||||
|
/* SizeOfHeaders is the combined size of an MS-DOS stub, PE header,
|
||||||
|
* and section headers rounded up to a multiple of FileAlignment. */
|
||||||
|
hdrsize = GET_UINT32_LE(indata + header_size + 84);
|
||||||
|
if (hdrsize < header_size || hdrsize > UINT32_MAX) {
|
||||||
|
printf("Corrupted headers size: 0x%08X\n", hdrsize);
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
}
|
||||||
|
/* SizeOfOptionalHeader is she size of the optional header, which is
|
||||||
|
* required for executable files, but for object files should be zero. */
|
||||||
|
opthdr_size = GET_UINT16_LE(indata + header_size + 20);
|
||||||
|
if (opthdr_size == 0 || opthdr_size > UINT16_MAX) {
|
||||||
|
printf("Corrupted optional header size: 0x%08X\n", opthdr_size);
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
}
|
||||||
|
pphlen = 4 + EVP_MD_size(md);
|
||||||
|
phlen = pphlen * (3 + (int)nsections + (int)(sigpos / pagesize));
|
||||||
|
res = OPENSSL_malloc((size_t)phlen);
|
||||||
|
zeroes = OPENSSL_zalloc((size_t)pagesize);
|
||||||
|
|
||||||
|
mdctx = EVP_MD_CTX_new();
|
||||||
if (!EVP_DigestInit(mdctx, md)) {
|
if (!EVP_DigestInit(mdctx, md)) {
|
||||||
EVP_MD_CTX_free(mdctx);
|
EVP_MD_CTX_free(mdctx);
|
||||||
printf("Unable to set up the digest context\n");
|
printf("Unable to set up the digest context\n");
|
||||||
return NULL; /* FAILED */
|
return NULL; /* FAILED */
|
||||||
}
|
}
|
||||||
nsections = GET_UINT16_LE(indata + header_size + 6);
|
|
||||||
pagesize = GET_UINT32_LE(indata + header_size + 56);
|
|
||||||
hdrsize = GET_UINT32_LE(indata + header_size + 84);
|
|
||||||
pphlen = 4 + EVP_MD_size(md);
|
|
||||||
phlen = pphlen * (3 + (int)nsections + (int)(sigpos / pagesize));
|
|
||||||
|
|
||||||
res = OPENSSL_malloc((size_t)phlen);
|
|
||||||
zeroes = OPENSSL_zalloc((size_t)pagesize);
|
|
||||||
|
|
||||||
EVP_DigestUpdate(mdctx, indata, header_size + 88);
|
EVP_DigestUpdate(mdctx, indata, header_size + 88);
|
||||||
EVP_DigestUpdate(mdctx, indata + header_size + 92, 60 + pe32plus*16);
|
EVP_DigestUpdate(mdctx, indata + header_size + 92, 60 + pe32plus*16);
|
||||||
EVP_DigestUpdate(mdctx, indata + header_size + 160 + pe32plus*16,
|
EVP_DigestUpdate(mdctx, indata + header_size + 160 + pe32plus*16,
|
||||||
@ -1591,9 +1626,9 @@ static u_char *pe_calc_page_hash(char *indata, uint32_t header_size,
|
|||||||
memset(res, 0, 4);
|
memset(res, 0, 4);
|
||||||
EVP_DigestFinal(mdctx, res + 4, NULL);
|
EVP_DigestFinal(mdctx, res + 4, NULL);
|
||||||
|
|
||||||
opthdr_size = GET_UINT16_LE(indata + header_size + 20);
|
|
||||||
sections = indata + header_size + 24 + opthdr_size;
|
sections = indata + header_size + 24 + opthdr_size;
|
||||||
for (i=0; i<nsections; i++) {
|
for (i=0; i<nsections; i++) {
|
||||||
|
/* Resource Table address and size */
|
||||||
rs = GET_UINT32_LE(sections + 16);
|
rs = GET_UINT32_LE(sections + 16);
|
||||||
ro = GET_UINT32_LE(sections + 20);
|
ro = GET_UINT32_LE(sections + 20);
|
||||||
if (rs == 0 || rs >= UINT32_MAX) {
|
if (rs == 0 || rs >= UINT32_MAX) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user