simplify MSI parsing recursion

This commit is contained in:
Michał Trojnara 2022-02-22 10:08:15 +01:00
parent f0207411b9
commit 95c5a4451b

110
msi.c
View File

@ -15,6 +15,8 @@
#define MIN(a,b) ((a) < (b) ? a : b) #define MIN(a,b) ((a) < (b) ? a : b)
static int recurse_entry(MSI_FILE *msi, uint32_t entryID, MSI_DIRENT *parent, MSI_DIRENT *prev, int verbose);
/* Get absolute address from sector and offset */ /* Get absolute address from sector and offset */
static const u_char *sector_offset_to_address(MSI_FILE *msi, uint32_t sector, uint32_t offset, int verbose) static const u_char *sector_offset_to_address(MSI_FILE *msi, uint32_t sector, uint32_t offset, int verbose)
{ {
@ -345,57 +347,43 @@ static MSI_ENTRY *parse_entry(MSI_FILE *msi, const u_char *data, int verbose)
* Pass "0" to get the root directory entry. -- This is the start point to navigate the compound file. * Pass "0" to get the root directory entry. -- This is the start point to navigate the compound file.
* Use the returned object to access child entries. * Use the returned object to access child entries.
*/ */
static int get_entry(MSI_FILE *msi, uint32_t entryID, int is_root, MSI_ENTRY **entry, int verbose) static MSI_ENTRY *get_entry(MSI_FILE *msi, uint32_t entryID, int is_root, int verbose)
{ {
uint32_t sector = 0; uint32_t sector = 0;
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) {
if (verbose) printf("Corrupted entryID\n"); if (verbose) printf("Corrupted entryID\n");
return 0; /* FAILED */ return NULL; /* FAILED */
} }
if (msi->m_bufferLen / sizeof(MSI_ENTRY) <= entryID) { if (msi->m_bufferLen / sizeof(MSI_ENTRY) <= entryID) {
if (verbose) printf("Invalid argument entryID\n"); if (verbose) printf("Invalid argument entryID\n");
return 0; /* FAILED */ return NULL; /* FAILED */
} }
/* The first entry in the first sector of the directory chain is known as /* 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 */ the root directory entry so it can not contain the directory stream */
if (msi->m_hdr->firstDirectorySectorLocation == 0 && entryID == 0) { if (msi->m_hdr->firstDirectorySectorLocation == 0 && entryID == 0) {
if (verbose) printf("Corrupted First Directory Sector Location\n"); if (verbose) printf("Corrupted First Directory Sector Location\n");
return 0; /* FAILED */ return NULL; /* FAILED */
} }
if (!locate_final_sector(msi, msi->m_hdr->firstDirectorySectorLocation, if (!locate_final_sector(msi, msi->m_hdr->firstDirectorySectorLocation,
entryID * sizeof(MSI_ENTRY), &sector, &offset, verbose)) { entryID * sizeof(MSI_ENTRY), &sector, &offset, verbose)) {
if (verbose) printf("Failed to locate a final sector\n"); if (verbose) printf("Failed to locate a final sector\n");
return 0; /* FAILED */ return NULL; /* FAILED */
} }
address = sector_offset_to_address(msi, sector, offset, verbose); address = sector_offset_to_address(msi, sector, offset, verbose);
if (!address) { if (!address) {
if (verbose) printf("Failed to get a final address\n"); if (verbose) printf("Failed to get a final address\n");
return 0; /* FAILED */ return NULL; /* FAILED */
} }
*entry = parse_entry(msi, address, verbose); return parse_entry(msi, address, verbose);
if (!*entry) {
if (verbose) printf("Failed to parse MSI_ENTRY struct\n");
return 0; /* FAILED */
}
return 1; /* OK */
} }
MSI_ENTRY *msi_root_entry_get(MSI_FILE *msi, int verbose) MSI_ENTRY *msi_root_entry_get(MSI_FILE *msi, int verbose)
{ {
MSI_ENTRY *entry = NULL; return get_entry(msi, 0, TRUE, verbose);
if (!get_entry(msi, 0, TRUE, &entry, verbose)) {
return NULL;
}
return entry;
} }
/* Parse MSI_FILE struct */ /* Parse MSI_FILE struct */
@ -446,11 +434,10 @@ MSI_FILE *msi_file_new(char *buffer, uint32_t len, int verbose)
return msi; return msi;
} }
/* Recursively parse MSI_DIRENT struct */ /* Recursively create a tree of MSI_DIRENT structures */
int msi_dirent_new(MSI_FILE *msi, MSI_ENTRY *entry, MSI_DIRENT *parent, MSI_DIRENT *prev, MSI_DIRENT **ret, int verbose) int msi_dirent_new(MSI_FILE *msi, MSI_ENTRY *entry, MSI_DIRENT *parent, MSI_DIRENT *prev, MSI_DIRENT **ret, int verbose)
{ {
MSI_DIRENT *dirent; MSI_DIRENT *dirent;
MSI_ENTRY *lnode = NULL, *rnode = NULL, *cnode = NULL;
if (!entry) { if (!entry) {
return 1; /* OK */ return 1; /* OK */
@ -482,52 +469,41 @@ int msi_dirent_new(MSI_FILE *msi, MSI_ENTRY *entry, MSI_DIRENT *parent, MSI_DIRE
OPENSSL_free(dirent); OPENSSL_free(dirent);
return 0; /* FAILED */ return 0; /* FAILED */
} }
/* NOTE : These links are a tree, not a linked list */
/* The special value NOSTREAM (0xFFFFFFFF) is used as a terminator */ if (!recurse_entry(msi, entry->leftSiblingID, parent, dirent, verbose)
if (entry->leftSiblingID != NOSTREAM) { || !recurse_entry(msi, entry->rightSiblingID, parent, dirent, verbose)
if (!get_entry(msi, entry->leftSiblingID, FALSE, &lnode, verbose)) { || !recurse_entry(msi, entry->childID, dirent, dirent, verbose)) {
if (verbose) printf("Corrupted Left Sibling ID: 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, lnode, parent, dirent, NULL, verbose)) {
OPENSSL_free(lnode);
sk_MSI_DIRENT_free(dirent->children);
OPENSSL_free(dirent);
return 0; /* FAILED */
}
}
if (entry->rightSiblingID != NOSTREAM) {
if (!get_entry(msi, entry->rightSiblingID, FALSE, &rnode, verbose)) {
if (verbose) printf("Corrupted Right Sibling ID: 0x%08X\n", entry->rightSiblingID);
sk_MSI_DIRENT_free(dirent->children);
OPENSSL_free(dirent);
return 0; /* FAILED */
}
if (!msi_dirent_new(msi, rnode, parent, dirent, NULL, verbose)) {
OPENSSL_free(rnode);
sk_MSI_DIRENT_free(dirent->children);
OPENSSL_free(dirent);
return 0; /* FAILED */
}
}
if (entry->type != DIR_STREAM && entry->childID != NOSTREAM) {
if (!get_entry(msi, entry->childID, FALSE, &cnode, verbose)) {
if (verbose) printf("Corrupted Child ID: 0x%08X\n", entry->childID);
sk_MSI_DIRENT_free(dirent->children);
OPENSSL_free(dirent);
return 0; /* FAILED */
}
if (!msi_dirent_new(msi, cnode, dirent, dirent, NULL, verbose)) {
OPENSSL_free(cnode);
sk_MSI_DIRENT_free(dirent->children);
OPENSSL_free(dirent);
return 0; /* FAILED */
}
} }
if (ret) if (ret)
*ret = dirent; *ret = dirent;
return 1; /* OK */
}
/* Add a sibling or a child to the tree */
static int recurse_entry(MSI_FILE *msi, uint32_t entryID, MSI_DIRENT *parent, MSI_DIRENT *dirent, int verbose) {
/* NOTE : These links are a tree, not a linked list */
MSI_ENTRY *node;
/* The special value NOSTREAM (0xFFFFFFFF) is used as a terminator */
if (entryID == NOSTREAM) /* stop condition */
return 1; /* OK */
node = get_entry(msi, entryID, FALSE, verbose);
if (!node) {
if (verbose) printf("Corrupted ID: 0x%08X\n", entryID);
return 0; /* FAILED */
}
if (!msi_dirent_new(msi, node, parent, dirent, NULL, verbose)) {
OPENSSL_free(node);
return 0; /* FAILED */
}
return 1; /* OK */ return 1; /* OK */
} }