mirror of
https://github.com/mtrojnar/osslsigncode.git
synced 2025-04-05 09:08:04 -05:00
fix double free in msi_dirent_new()
This commit is contained in:
parent
bdea1d1c2a
commit
289c345280
53
msi.c
53
msi.c
@ -162,20 +162,28 @@ static uint32_t get_next_mini_sector(MSI_FILE *msi, uint32_t miniSector)
|
|||||||
return GET_UINT32_LE(address);
|
return GET_UINT32_LE(address);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void locate_final_mini_sector(MSI_FILE *msi, uint32_t sector, uint32_t offset, uint32_t *finalSector, uint32_t *finalOffset)
|
static int locate_final_mini_sector(MSI_FILE *msi, uint32_t sector, uint32_t offset, uint32_t *finalSector, uint32_t *finalOffset)
|
||||||
{
|
{
|
||||||
while (offset >= msi->m_minisectorSize) {
|
while (offset >= msi->m_minisectorSize) {
|
||||||
offset -= msi->m_minisectorSize;
|
offset -= msi->m_minisectorSize;
|
||||||
sector = get_next_mini_sector(msi, sector);
|
sector = get_next_mini_sector(msi, sector);
|
||||||
|
if (sector == 0) {
|
||||||
|
printf("Failed to get a next mini sector\n");
|
||||||
|
return 0; /* FAILED */
|
||||||
|
}
|
||||||
}
|
}
|
||||||
*finalSector = sector;
|
*finalSector = sector;
|
||||||
*finalOffset = offset;
|
*finalOffset = offset;
|
||||||
|
return 1; /* OK */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Same logic as "read_stream" except that use mini stream functions instead */
|
/* Same logic as "read_stream" except that use mini stream functions instead */
|
||||||
static int read_mini_stream(MSI_FILE *msi, uint32_t sector, uint32_t offset, char *buffer, uint32_t len)
|
static int read_mini_stream(MSI_FILE *msi, uint32_t sector, uint32_t offset, char *buffer, uint32_t len)
|
||||||
{
|
{
|
||||||
locate_final_mini_sector(msi, sector, offset, §or, &offset);
|
if (!locate_final_mini_sector(msi, sector, offset, §or, &offset)) {
|
||||||
|
printf("Failed to locate a final mini sector\n");
|
||||||
|
return 0; /* FAILED */
|
||||||
|
}
|
||||||
while (len > 0) {
|
while (len > 0) {
|
||||||
const u_char *address;
|
const u_char *address;
|
||||||
uint32_t copylen;
|
uint32_t copylen;
|
||||||
@ -193,6 +201,10 @@ static int read_mini_stream(MSI_FILE *msi, uint32_t sector, uint32_t offset, cha
|
|||||||
buffer += copylen;
|
buffer += copylen;
|
||||||
len -= copylen;
|
len -= copylen;
|
||||||
sector = get_next_mini_sector(msi, sector);
|
sector = get_next_mini_sector(msi, sector);
|
||||||
|
if (sector == 0) {
|
||||||
|
printf("Failed to get a next mini sector\n");
|
||||||
|
return 0; /* FAILED */
|
||||||
|
}
|
||||||
offset = 0;
|
offset = 0;
|
||||||
}
|
}
|
||||||
return 1; /* OK */
|
return 1; /* OK */
|
||||||
@ -273,7 +285,8 @@ static MSI_ENTRY *parse_entry(MSI_FILE *msi, const u_char *data)
|
|||||||
return NULL; /* FAILED */
|
return NULL; /* FAILED */
|
||||||
}
|
}
|
||||||
memcpy(entry->size, data + DIRENT_FILE_SIZE, 8);
|
memcpy(entry->size, data + DIRENT_FILE_SIZE, 8);
|
||||||
/* For a version 3 compound file 512-byte sector size, the value of this field MUST be less than or equal to 0x80000000 */
|
/* For a version 3 compound file 512-byte sector size, the value of this field
|
||||||
|
MUST be less than or equal to 0x80000000 */
|
||||||
inlen = GET_UINT32_LE(entry->size);
|
inlen = GET_UINT32_LE(entry->size);
|
||||||
if ((msi->m_sectorSize == 0x0200 && inlen > 0x80000000)
|
if ((msi->m_sectorSize == 0x0200 && inlen > 0x80000000)
|
||||||
|| (msi->m_bufferLen <= inlen)) {
|
|| (msi->m_bufferLen <= inlen)) {
|
||||||
@ -295,19 +308,25 @@ static int get_entry(MSI_FILE *msi, uint32_t entryID, int is_root, MSI_ENTRY **e
|
|||||||
uint32_t offset = 0;
|
uint32_t offset = 0;
|
||||||
const u_char *address;
|
const u_char *address;
|
||||||
|
|
||||||
|
/* The special value NOSTREAM (0xFFFFFFFF) is used as a terminator */
|
||||||
|
if (entryID == NOSTREAM) {
|
||||||
|
return 1; /* OK */
|
||||||
|
}
|
||||||
/* Corrupted file */
|
/* Corrupted file */
|
||||||
if (!is_root && entryID == 0) {
|
if (!is_root && entryID == 0) {
|
||||||
printf("Corrupted file\n");
|
printf("Corrupted file\n");
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
}
|
}
|
||||||
/* The special value NOSTREAM (0xFFFFFFFF) is used as a terminator */
|
|
||||||
if (entryID == NOSTREAM) {
|
|
||||||
return 1; /* OK */
|
|
||||||
}
|
|
||||||
if (msi->m_bufferLen / sizeof(MSI_ENTRY) <= entryID) {
|
if (msi->m_bufferLen / sizeof(MSI_ENTRY) <= entryID) {
|
||||||
printf("Invalid argument entryID\n");
|
printf("Invalid argument entryID\n");
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
}
|
}
|
||||||
|
/* The first entry in the first sector of the directory chain is known as
|
||||||
|
the root directory entry so it can not contain the directory stream */
|
||||||
|
if (msi->m_hdr->firstDirectorySectorLocation == 0 && entryID == 0) {
|
||||||
|
printf("Corrupted First Directory Sector Location\n");
|
||||||
|
return 0; /* FAILED */
|
||||||
|
}
|
||||||
if (!locate_final_sector(msi, msi->m_hdr->firstDirectorySectorLocation, entryID * sizeof(MSI_ENTRY), §or, &offset)) {
|
if (!locate_final_sector(msi, msi->m_hdr->firstDirectorySectorLocation, entryID * sizeof(MSI_ENTRY), §or, &offset)) {
|
||||||
printf("Failed to locate a final sector\n");
|
printf("Failed to locate a final sector\n");
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
@ -387,7 +406,7 @@ MSI_FILE_HDR *msi_header_get(MSI_FILE *msi)
|
|||||||
int msi_dirent_new(MSI_FILE *msi, MSI_ENTRY *entry, MSI_DIRENT *parent, MSI_DIRENT **ret)
|
int msi_dirent_new(MSI_FILE *msi, MSI_ENTRY *entry, MSI_DIRENT *parent, MSI_DIRENT **ret)
|
||||||
{
|
{
|
||||||
MSI_DIRENT *dirent, *unused = NULL;
|
MSI_DIRENT *dirent, *unused = NULL;
|
||||||
MSI_ENTRY *node = NULL;
|
MSI_ENTRY *lnode = NULL, *rnode = NULL, *cnode = NULL;
|
||||||
|
|
||||||
if (!entry) {
|
if (!entry) {
|
||||||
return 1; /* OK */
|
return 1; /* OK */
|
||||||
@ -410,39 +429,39 @@ int msi_dirent_new(MSI_FILE *msi, MSI_ENTRY *entry, MSI_DIRENT *parent, MSI_DIRE
|
|||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
}
|
}
|
||||||
/* NOTE : These links are a tree, not a linked list */
|
/* NOTE : These links are a tree, not a linked list */
|
||||||
if (!get_entry(msi, entry->leftSiblingID, FALSE, &node)) {
|
if (!get_entry(msi, entry->leftSiblingID, FALSE, &lnode)) {
|
||||||
printf("Corrupted leftSiblingID: 0x%08X\n", entry->leftSiblingID);
|
printf("Corrupted leftSiblingID: 0x%08X\n", entry->leftSiblingID);
|
||||||
sk_MSI_DIRENT_free(dirent->children);
|
sk_MSI_DIRENT_free(dirent->children);
|
||||||
OPENSSL_free(dirent);
|
OPENSSL_free(dirent);
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
}
|
}
|
||||||
if (!msi_dirent_new(msi, node, parent, &unused)) {
|
if (!msi_dirent_new(msi, lnode, parent, &unused)) {
|
||||||
OPENSSL_free(node);
|
OPENSSL_free(lnode);
|
||||||
sk_MSI_DIRENT_free(dirent->children);
|
sk_MSI_DIRENT_free(dirent->children);
|
||||||
OPENSSL_free(dirent);
|
OPENSSL_free(dirent);
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
}
|
}
|
||||||
if (!get_entry(msi, entry->rightSiblingID, FALSE, &node)) {
|
if (!get_entry(msi, entry->rightSiblingID, FALSE, &rnode)) {
|
||||||
printf("Corrupted rightSiblingID: 0x%08X\n", entry->rightSiblingID);
|
printf("Corrupted rightSiblingID: 0x%08X\n", entry->rightSiblingID);
|
||||||
sk_MSI_DIRENT_free(dirent->children);
|
sk_MSI_DIRENT_free(dirent->children);
|
||||||
OPENSSL_free(dirent);
|
OPENSSL_free(dirent);
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
}
|
}
|
||||||
if (!msi_dirent_new(msi, node, parent, &unused)) {
|
if (!msi_dirent_new(msi, rnode, parent, &unused)) {
|
||||||
OPENSSL_free(node);
|
OPENSSL_free(rnode);
|
||||||
sk_MSI_DIRENT_free(dirent->children);
|
sk_MSI_DIRENT_free(dirent->children);
|
||||||
OPENSSL_free(dirent);
|
OPENSSL_free(dirent);
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
}
|
}
|
||||||
if (entry->type != DIR_STREAM) {
|
if (entry->type != DIR_STREAM) {
|
||||||
if (!get_entry(msi, entry->childID, FALSE, &node)) {
|
if (!get_entry(msi, entry->childID, FALSE, &cnode)) {
|
||||||
printf("Corrupted childID: 0x%08X\n", entry->childID);
|
printf("Corrupted childID: 0x%08X\n", entry->childID);
|
||||||
sk_MSI_DIRENT_free(dirent->children);
|
sk_MSI_DIRENT_free(dirent->children);
|
||||||
OPENSSL_free(dirent);
|
OPENSSL_free(dirent);
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
}
|
}
|
||||||
if (!msi_dirent_new(msi, node, dirent, &unused)) {
|
if (!msi_dirent_new(msi, cnode, dirent, &unused)) {
|
||||||
OPENSSL_free(node);
|
OPENSSL_free(cnode);
|
||||||
sk_MSI_DIRENT_free(dirent->children);
|
sk_MSI_DIRENT_free(dirent->children);
|
||||||
OPENSSL_free(dirent);
|
OPENSSL_free(dirent);
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user