mirror of
https://github.com/mtrojnar/osslsigncode.git
synced 2025-04-05 17:08:05 -05:00
verify msi metadata
This commit is contained in:
parent
315357f092
commit
6df4c12624
351
osslsigncode.c
351
osslsigncode.c
@ -102,6 +102,7 @@ typedef unsigned char u_char;
|
|||||||
#ifdef WITH_GSF
|
#ifdef WITH_GSF
|
||||||
#include <gsf/gsf-infile-msole.h>
|
#include <gsf/gsf-infile-msole.h>
|
||||||
#include <gsf/gsf-infile.h>
|
#include <gsf/gsf-infile.h>
|
||||||
|
#include <gsf/gsf-infile-impl.h>
|
||||||
#include <gsf/gsf-input-stdio.h>
|
#include <gsf/gsf-input-stdio.h>
|
||||||
#include <gsf/gsf-outfile-msole.h>
|
#include <gsf/gsf-outfile-msole.h>
|
||||||
#include <gsf/gsf-outfile.h>
|
#include <gsf/gsf-outfile.h>
|
||||||
@ -290,12 +291,83 @@ typedef struct {
|
|||||||
} CRYPTO_PARAMS;
|
} CRYPTO_PARAMS;
|
||||||
|
|
||||||
#ifdef WITH_GSF
|
#ifdef WITH_GSF
|
||||||
|
#define OLE_HEADER_SIZE 0x200 /* independent of big block size */
|
||||||
|
#define DIRENT_MAX_NAME_SIZE 0x40
|
||||||
|
#define DIRENT_DETAILS_SIZE 0x40
|
||||||
|
#define DIRENT_SIZE (DIRENT_MAX_NAME_SIZE + DIRENT_DETAILS_SIZE)
|
||||||
|
#define DIRENT_NAME_LEN 0x40 /* length in bytes incl 0 terminator */
|
||||||
|
#define DIRENT_TYPE 0x42
|
||||||
|
#define DIRENT_PREV 0x44
|
||||||
|
#define DIRENT_NEXT 0x48
|
||||||
|
#define DIRENT_CHILD 0x4c
|
||||||
|
#define DIRENT_CLSID 0x50
|
||||||
|
#define DIRENT_USERFLAGS 0x60
|
||||||
|
#define DIRENT_CREATE_TIME 0x64
|
||||||
|
#define DIRENT_MODIFY_TIME 0x6c
|
||||||
|
#define DIRENT_FILE_SIZE 0x78
|
||||||
|
#define DIRENT_MAGIC_END 0xffffffff
|
||||||
|
|
||||||
|
#define DIRENT_TYPE_STORAGE 1
|
||||||
|
#define DIRENT_TYPE_STREAM 2
|
||||||
|
#define DIRENT_TYPE_ROOT 5
|
||||||
|
|
||||||
|
#define OLE_BIG_BLOCK(index, ole) ((index) >> ole->info->bb.shift)
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
GsfOutfile *outole;
|
GsfOutfile *outole;
|
||||||
GsfOutput *sink;
|
GsfOutput *sink;
|
||||||
u_char *p_msiex;
|
u_char *p_msiex;
|
||||||
int len_msiex;
|
int len_msiex;
|
||||||
} GSF_PARAMS;
|
} GSF_PARAMS;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
guint32 *block;
|
||||||
|
guint32 num_blocks;
|
||||||
|
} MSOleBAT;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int entry;
|
||||||
|
guint16 name_len;
|
||||||
|
guint8 type;
|
||||||
|
guint32 prev;
|
||||||
|
guint32 next;
|
||||||
|
guint32 child;
|
||||||
|
unsigned char clsid[16];
|
||||||
|
unsigned char size[4];
|
||||||
|
unsigned char flags[4];
|
||||||
|
unsigned char cretime[8];
|
||||||
|
unsigned char modtime[8];
|
||||||
|
unsigned char name[DIRENT_MAX_NAME_SIZE];
|
||||||
|
GSList *children;
|
||||||
|
} MSOleDirent;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
struct {
|
||||||
|
MSOleBAT bat;
|
||||||
|
unsigned shift;
|
||||||
|
unsigned filter;
|
||||||
|
size_t size;
|
||||||
|
} bb, sb;
|
||||||
|
gsf_off_t max_block;
|
||||||
|
guint32 threshold;
|
||||||
|
guint32 sbat_start, num_sbat;
|
||||||
|
MSOleDirent *root_dir;
|
||||||
|
GsfInput *sb_file;
|
||||||
|
int ref_count;
|
||||||
|
} MSOleInfo;
|
||||||
|
|
||||||
|
struct _GsfInfileMSOle {
|
||||||
|
GsfInfile parent;
|
||||||
|
GsfInput *input;
|
||||||
|
MSOleInfo *info;
|
||||||
|
MSOleDirent *dirent;
|
||||||
|
MSOleBAT bat;
|
||||||
|
gsf_off_t cur_block;
|
||||||
|
struct {
|
||||||
|
guint8 *buf;
|
||||||
|
size_t buf_size;
|
||||||
|
} stream;
|
||||||
|
};
|
||||||
#endif /* WITH_GSF */
|
#endif /* WITH_GSF */
|
||||||
|
|
||||||
|
|
||||||
@ -680,6 +752,25 @@ ASN1_SEQUENCE(TimeStampToken) = {
|
|||||||
|
|
||||||
IMPLEMENT_ASN1_FUNCTIONS(TimeStampToken)
|
IMPLEMENT_ASN1_FUNCTIONS(TimeStampToken)
|
||||||
|
|
||||||
|
#ifdef WITH_GSF
|
||||||
|
static const u_char digital_signature[] = {
|
||||||
|
0x05, 0x00, 0x44, 0x00, 0x69, 0x00, 0x67, 0x00,
|
||||||
|
0x69, 0x00, 0x74, 0x00, 0x61, 0x00, 0x6C, 0x00,
|
||||||
|
0x53, 0x00, 0x69, 0x00, 0x67, 0x00, 0x6E, 0x00,
|
||||||
|
0x61, 0x00, 0x74, 0x00, 0x75, 0x00, 0x72, 0x00,
|
||||||
|
0x65, 0x00
|
||||||
|
};
|
||||||
|
|
||||||
|
static const u_char digital_signature_ex[] = {
|
||||||
|
0x05, 0x00, 0x4D, 0x00, 0x73, 0x00, 0x69, 0x00,
|
||||||
|
0x44, 0x00, 0x69, 0x00, 0x67, 0x00, 0x69, 0x00,
|
||||||
|
0x74, 0x00, 0x61, 0x00, 0x6C, 0x00, 0x53, 0x00,
|
||||||
|
0x69, 0x00, 0x67, 0x00, 0x6E, 0x00, 0x61, 0x00,
|
||||||
|
0x74, 0x00, 0x75, 0x00, 0x72, 0x00, 0x65, 0x00,
|
||||||
|
0x45, 0x00, 0x78, 0x00
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* $ echo -n 3006030200013000 | xxd -r -p | openssl asn1parse -i -inform der
|
* $ echo -n 3006030200013000 | xxd -r -p | openssl asn1parse -i -inform der
|
||||||
* 0:d=0 hl=2 l= 6 cons: SEQUENCE
|
* 0:d=0 hl=2 l= 6 cons: SEQUENCE
|
||||||
@ -1389,7 +1480,7 @@ static void help_for(const char *argv0, const char *cmd)
|
|||||||
if (on_list(cmd, cmds_st))
|
if (on_list(cmd, cmds_st))
|
||||||
printf("%-24s= the unix-time to set the signing time\n", "-st");
|
printf("%-24s= the unix-time to set the signing time\n", "-st");
|
||||||
if (on_list(cmd, cmds_timestamp_expiration))
|
if (on_list(cmd, cmds_timestamp_expiration))
|
||||||
printf("%-24s= verify a finite lifetime of the TSA private key\n", "-st");
|
printf("%-24s= verify a finite lifetime of the TSA private key\n", "-timestamp-expiration");
|
||||||
#ifdef ENABLE_CURL
|
#ifdef ENABLE_CURL
|
||||||
if (on_list(cmd, cmds_t)) {
|
if (on_list(cmd, cmds_t)) {
|
||||||
printf("%-24s= specifies that the digital signature will be timestamped\n", "-t");
|
printf("%-24s= specifies that the digital signature will be timestamped\n", "-t");
|
||||||
@ -2861,6 +2952,7 @@ static int verify_signature(SIGNATURE *signature, GLOBAL_OPTIONS *options)
|
|||||||
#ifdef WITH_GSF
|
#ifdef WITH_GSF
|
||||||
/*
|
/*
|
||||||
* MSI file support
|
* MSI file support
|
||||||
|
* https://msdn.microsoft.com/en-us/library/dd942138.aspx
|
||||||
*/
|
*/
|
||||||
static gint msi_base64_decode(gint x)
|
static gint msi_base64_decode(gint x)
|
||||||
{
|
{
|
||||||
@ -2958,23 +3050,90 @@ static GSList *msi_sorted_infile_children(GsfInfile *infile)
|
|||||||
return sorted;
|
return sorted;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
static guint8 const *msi_get_block(GsfInfileMSOle const *ole, guint32 entry)
|
||||||
* msi_prehash_utf16_name converts an UTF-8 representation of
|
|
||||||
* an MSI filename to its on-disk UTF-16 representation and
|
|
||||||
* writes it to the hash BIO. It is used when calculating the
|
|
||||||
* pre-hash used for MsiDigitalSignatureEx signatures in MSI files.
|
|
||||||
*/
|
|
||||||
static gboolean msi_prehash_utf16_name(gchar *name, BIO *hash)
|
|
||||||
{
|
{
|
||||||
glong chars_written = 0;
|
guint8 const *data;
|
||||||
|
guint32 block, oleblock;
|
||||||
|
|
||||||
gchar *u16name = (gchar*)g_utf8_to_utf16(name, -1, NULL, &chars_written, NULL);
|
if (entry >= DIRENT_MAGIC_END) {
|
||||||
if (u16name == NULL) {
|
return NULL; /* FAILED */
|
||||||
return FALSE; /* FAILED */
|
|
||||||
}
|
}
|
||||||
BIO_write(hash, u16name, 2*chars_written);
|
if (entry > G_MAXUINT / DIRENT_SIZE) {
|
||||||
g_free(u16name);
|
return NULL; /* FAILED */
|
||||||
return TRUE;
|
}
|
||||||
|
block = OLE_BIG_BLOCK(entry * DIRENT_SIZE, ole);
|
||||||
|
oleblock = ole->bat.block[block];
|
||||||
|
if (oleblock >= ole->info->max_block)
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
|
||||||
|
/* OLE_HEADER_SIZE is fixed at 512, but the sector containing the
|
||||||
|
* header is padded out to bb.size (sector size) when bb.size > 512. */
|
||||||
|
if (gsf_input_seek(ole->input, (gsf_off_t)(MAX(OLE_HEADER_SIZE, ole->info->bb.size)
|
||||||
|
+ (oleblock << ole->info->bb.shift)), G_SEEK_SET)) {
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
}
|
||||||
|
data = gsf_input_read(ole->input, ole->info->bb.size, NULL);
|
||||||
|
if (data == NULL) {
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
}
|
||||||
|
data += (DIRENT_SIZE * entry) % ole->info->bb.size;
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gint msi_dirent_cmp(MSOleDirent const *a, MSOleDirent const *b)
|
||||||
|
{
|
||||||
|
gint diff = memcmp(a->name, b->name, MIN(a->name_len, b->name_len));
|
||||||
|
/* apparently the longer wins */
|
||||||
|
if (diff == 0) {
|
||||||
|
return a->name_len > b->name_len ? 1 : -1;
|
||||||
|
}
|
||||||
|
return diff;
|
||||||
|
}
|
||||||
|
|
||||||
|
MSOleDirent *msi_dirent_new(GsfInfileMSOle *ole, guint32 entry, MSOleDirent *parent)
|
||||||
|
{
|
||||||
|
MSOleDirent *dirent;
|
||||||
|
guint8 const *data;
|
||||||
|
guint8 type;
|
||||||
|
|
||||||
|
data = msi_get_block(ole, entry);
|
||||||
|
if (data == NULL) {
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
}
|
||||||
|
type = GSF_LE_GET_GUINT8(data + DIRENT_TYPE);
|
||||||
|
if (type != DIRENT_TYPE_STORAGE && type != DIRENT_TYPE_STREAM && type != DIRENT_TYPE_ROOT) {
|
||||||
|
printf("Unknown stream type 0x%02x\n", type);
|
||||||
|
return NULL; /* FAILED */
|
||||||
|
}
|
||||||
|
|
||||||
|
dirent = g_new0(MSOleDirent, 1);
|
||||||
|
dirent->entry = entry;
|
||||||
|
dirent->name_len = GSF_LE_GET_GUINT16(data + DIRENT_NAME_LEN) - 2;
|
||||||
|
dirent->type = type;
|
||||||
|
dirent->prev = GSF_LE_GET_GUINT32(data + DIRENT_PREV);
|
||||||
|
dirent->next = GSF_LE_GET_GUINT32(data + DIRENT_NEXT);
|
||||||
|
dirent->child = GSF_LE_GET_GUINT32(data + DIRENT_CHILD);
|
||||||
|
memcpy(dirent->clsid, data + DIRENT_CLSID, sizeof(dirent->clsid));
|
||||||
|
memcpy(dirent->size, data + DIRENT_FILE_SIZE, sizeof(dirent->size));
|
||||||
|
memcpy(dirent->flags, data + DIRENT_USERFLAGS, sizeof(dirent->flags));
|
||||||
|
memcpy(dirent->cretime, data + DIRENT_CREATE_TIME, sizeof(dirent->cretime));
|
||||||
|
memcpy(dirent->modtime, data + DIRENT_MODIFY_TIME, sizeof(dirent->modtime));
|
||||||
|
memcpy(dirent->name, data, dirent->name_len);
|
||||||
|
dirent->children = NULL;
|
||||||
|
|
||||||
|
if (parent != NULL) {
|
||||||
|
parent->children = g_slist_insert_sorted(parent->children, dirent, (GCompareFunc)msi_dirent_cmp);
|
||||||
|
}
|
||||||
|
/* NOTE : These links are a tree, not a linked list */
|
||||||
|
msi_dirent_new(ole, dirent->prev, parent);
|
||||||
|
msi_dirent_new(ole, dirent->next, parent);
|
||||||
|
|
||||||
|
if (dirent->type != DIRENT_TYPE_STREAM) {
|
||||||
|
msi_dirent_new(ole, dirent->child, dirent);
|
||||||
|
}
|
||||||
|
|
||||||
|
return dirent;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2985,78 +3144,72 @@ static gboolean msi_prehash_utf16_name(gchar *name, BIO *hash)
|
|||||||
*
|
*
|
||||||
* The hash is written to the hash BIO.
|
* The hash is written to the hash BIO.
|
||||||
*/
|
*/
|
||||||
static gboolean msi_prehash(GsfInfile *infile, gchar *dirname, BIO *hash)
|
|
||||||
|
/* Hash a MSI stream's extended metadata */
|
||||||
|
static void msi_prehash(MSOleDirent *dirent, BIO *hash)
|
||||||
{
|
{
|
||||||
GSList *sorted, *current;
|
if (dirent->type != DIRENT_TYPE_ROOT) {
|
||||||
guint8 classid[16], zeroes[8];
|
BIO_write(hash, dirent->name, dirent->name_len);
|
||||||
gboolean is_dir;
|
}
|
||||||
gsf_off_t size;
|
if (dirent->type != DIRENT_TYPE_STREAM) {
|
||||||
guint32 sizebuf;
|
BIO_write(hash, dirent->clsid, sizeof(dirent->clsid));
|
||||||
|
} else {
|
||||||
|
BIO_write(hash, dirent->size, sizeof(dirent->size));
|
||||||
|
}
|
||||||
|
BIO_write(hash, dirent->flags, sizeof(dirent->flags));
|
||||||
|
|
||||||
|
if (dirent->type != DIRENT_TYPE_ROOT) {
|
||||||
|
BIO_write(hash, dirent->cretime, sizeof(dirent->cretime));
|
||||||
|
BIO_write(hash, dirent->modtime, sizeof(dirent->modtime));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Recursively hash a MSI directory's extended metadata */
|
||||||
|
static gboolean msi_prehash_dir(MSOleDirent *dirent, BIO *hash)
|
||||||
|
{
|
||||||
|
GSList *tmp;
|
||||||
bool ret = FALSE;
|
bool ret = FALSE;
|
||||||
|
|
||||||
memset(&zeroes, 0, sizeof(zeroes));
|
if (dirent == NULL) {
|
||||||
gsf_infile_msole_get_class_id(GSF_INFILE_MSOLE(infile), classid);
|
goto out;
|
||||||
|
|
||||||
if (dirname != NULL) {
|
|
||||||
if (!msi_prehash_utf16_name(dirname, hash))
|
|
||||||
return ret; /* FAILED */
|
|
||||||
}
|
}
|
||||||
BIO_write(hash, classid, sizeof(classid));
|
msi_prehash(dirent, hash);
|
||||||
BIO_write(hash, zeroes, 4);
|
for (tmp = dirent->children; tmp; tmp = tmp->next) {
|
||||||
if (dirname != NULL) {
|
MSOleDirent *child = tmp->data;
|
||||||
/*
|
if (!memcmp(child->name, digital_signature, sizeof(digital_signature))
|
||||||
* Creation time and modification time for the root directory.
|
|| !memcmp(child->name, digital_signature_ex, sizeof(digital_signature_ex))) {
|
||||||
* These are always zero. The ctime and mtime of the actual
|
|
||||||
* file itself takes precedence.
|
|
||||||
*/
|
|
||||||
BIO_write(hash, zeroes, 8); /* ctime as Windows FILETIME */
|
|
||||||
BIO_write(hash, zeroes, 8); /* mtime as Windows FILETIME */
|
|
||||||
}
|
|
||||||
sorted = msi_sorted_infile_children(infile);
|
|
||||||
for (current = sorted; current; current = g_slist_next(current)) {
|
|
||||||
gchar *name = current->data;
|
|
||||||
GsfInput *child = gsf_infile_child_by_name(infile, name);
|
|
||||||
if (child == NULL)
|
|
||||||
continue;
|
continue;
|
||||||
is_dir = GSF_IS_INFILE(child) && gsf_infile_num_children(GSF_INFILE(child)) > 0;
|
}
|
||||||
if (is_dir) {
|
if (child->type == DIRENT_TYPE_STREAM) {
|
||||||
if (!msi_prehash(GSF_INFILE(child), name, hash)) {
|
msi_prehash(child, hash);
|
||||||
g_object_unref(child);
|
}
|
||||||
|
if (child->type == DIRENT_TYPE_STORAGE) {
|
||||||
|
if (!msi_prehash_dir(child, hash)) {
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
if (!msi_prehash_utf16_name(name, hash)) {
|
|
||||||
g_object_unref(child);
|
|
||||||
goto out;
|
|
||||||
}
|
}
|
||||||
/*
|
|
||||||
* File size.
|
|
||||||
*/
|
|
||||||
size = gsf_input_remaining(child);
|
|
||||||
sizebuf = GUINT32_TO_LE((guint32)size);
|
|
||||||
BIO_write(hash, &sizebuf, sizeof(sizebuf));
|
|
||||||
/*
|
|
||||||
* Reserved - must be 0. Corresponds to
|
|
||||||
* offset 0x7c..0x7f in the CDFv2 file.
|
|
||||||
*/
|
|
||||||
BIO_write(hash, zeroes, 4);
|
|
||||||
/*
|
|
||||||
* Creation time and modification time
|
|
||||||
* as Windows FILETIMEs. We keep them
|
|
||||||
* zeroed, because libgsf doesn't seem
|
|
||||||
* to support outputting them.
|
|
||||||
*/
|
|
||||||
BIO_write(hash, zeroes, 8); /* ctime as Windows FILETIME */
|
|
||||||
BIO_write(hash, zeroes, 8); /* mtime as Windows FILETIME */
|
|
||||||
}
|
|
||||||
g_object_unref(child);
|
|
||||||
}
|
}
|
||||||
ret = TRUE;
|
ret = TRUE;
|
||||||
out:
|
out:
|
||||||
g_slist_free_full(sorted, g_free);
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean msi_dirent_free(MSOleDirent *dirent)
|
||||||
|
{
|
||||||
|
GSList *tmp;
|
||||||
|
|
||||||
|
if (dirent == NULL) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
for (tmp = dirent->children; tmp; tmp = tmp->next) {
|
||||||
|
MSOleDirent *child = tmp->data;
|
||||||
|
msi_dirent_free(child);
|
||||||
|
}
|
||||||
|
g_slist_free(dirent->children);
|
||||||
|
g_free(dirent);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* msi_handle_dir performs a direct copy of the input MSI file in infile to a new
|
* msi_handle_dir performs a direct copy of the input MSI file in infile to a new
|
||||||
* output file in outfile. While copying, it also writes all file content to the
|
* output file in outfile. While copying, it also writes all file content to the
|
||||||
@ -3087,8 +3240,9 @@ static gboolean msi_handle_dir(GsfInfile *infile, GsfOutfile *outole, BIO *hash)
|
|||||||
if (child == NULL)
|
if (child == NULL)
|
||||||
continue;
|
continue;
|
||||||
is_dir = GSF_IS_INFILE(child) && gsf_infile_num_children(GSF_INFILE(child)) > 0;
|
is_dir = GSF_IS_INFILE(child) && gsf_infile_num_children(GSF_INFILE(child)) > 0;
|
||||||
if (outole != NULL)
|
if (outole != NULL) {
|
||||||
outchild = gsf_outfile_new_child(outole, name, is_dir);
|
outchild = gsf_outfile_new_child(outole, name, is_dir);
|
||||||
|
}
|
||||||
if (is_dir) {
|
if (is_dir) {
|
||||||
if (!msi_handle_dir(GSF_INFILE(child), GSF_OUTFILE(outchild), hash)) {
|
if (!msi_handle_dir(GSF_INFILE(child), GSF_OUTFILE(outchild), hash)) {
|
||||||
gsf_output_close(outchild);
|
gsf_output_close(outchild);
|
||||||
@ -3121,13 +3275,6 @@ out:
|
|||||||
g_slist_free_full(sorted, g_free);
|
g_slist_free_full(sorted, g_free);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Until libgsf can read more MSI metadata,
|
|
||||||
* we can't verify MsiDigitalSignatureEx
|
|
||||||
* #define GSF_CAN_READ_MSI_METADATA
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* msi_verify_pkcs7 is a helper function for msi_verify_file.
|
* msi_verify_pkcs7 is a helper function for msi_verify_file.
|
||||||
* It exists to make it easier to implement verification of nested signatures.
|
* It exists to make it easier to implement verification of nested signatures.
|
||||||
@ -3138,9 +3285,7 @@ static int msi_verify_pkcs7(SIGNATURE *signature, GsfInfile *infile, unsigned ch
|
|||||||
int ret = 1, mdtype = -1, mdok;
|
int ret = 1, mdtype = -1, mdok;
|
||||||
unsigned char mdbuf[EVP_MAX_MD_SIZE];
|
unsigned char mdbuf[EVP_MAX_MD_SIZE];
|
||||||
unsigned char cmdbuf[EVP_MAX_MD_SIZE];
|
unsigned char cmdbuf[EVP_MAX_MD_SIZE];
|
||||||
#ifdef GSF_CAN_READ_MSI_METADATA
|
|
||||||
unsigned char cexmdbuf[EVP_MAX_MD_SIZE];
|
unsigned char cexmdbuf[EVP_MAX_MD_SIZE];
|
||||||
#endif
|
|
||||||
char hexbuf[EVP_MAX_MD_SIZE*2+1];
|
char hexbuf[EVP_MAX_MD_SIZE*2+1];
|
||||||
const EVP_MD *md;
|
const EVP_MD *md;
|
||||||
BIO *hash;
|
BIO *hash;
|
||||||
@ -3167,35 +3312,23 @@ static int msi_verify_pkcs7(SIGNATURE *signature, GsfInfile *infile, unsigned ch
|
|||||||
BIO_set_md(hash, md);
|
BIO_set_md(hash, md);
|
||||||
BIO_push(hash, BIO_new(BIO_s_null()));
|
BIO_push(hash, BIO_new(BIO_s_null()));
|
||||||
if (exdata) {
|
if (exdata) {
|
||||||
/*
|
MSOleDirent *dirent;
|
||||||
* Until libgsf can read more MSI metadata, we can't
|
|
||||||
* really verify them by plowing through the file.
|
|
||||||
* Verifying files signed by osslsigncode itself works,
|
|
||||||
* though!
|
|
||||||
*
|
|
||||||
* For now, the compromise is to use the hash given
|
|
||||||
* by the file, which is equivalent to verifying a
|
|
||||||
* non-MsiDigitalSignatureEx signature from a security
|
|
||||||
* perspective, because we'll only be calculating the
|
|
||||||
* file content hashes ourselves.
|
|
||||||
*/
|
|
||||||
#ifdef GSF_CAN_READ_MSI_METADATA
|
|
||||||
BIO *prehash = BIO_new(BIO_f_md());
|
BIO *prehash = BIO_new(BIO_f_md());
|
||||||
BIO_set_md(prehash, md);
|
BIO_set_md(prehash, md);
|
||||||
BIO_push(prehash, BIO_new(BIO_s_null()));
|
BIO_push(prehash, BIO_new(BIO_s_null()));
|
||||||
|
|
||||||
if (!msi_prehash(infile, NULL, prehash)) {
|
dirent = msi_dirent_new(GSF_INFILE_MSOLE(infile), 0, NULL);
|
||||||
|
if (!msi_prehash_dir(dirent, prehash)) {
|
||||||
printf("Failed to calculate pre-hash used for MsiDigitalSignatureEx\n\n");
|
printf("Failed to calculate pre-hash used for MsiDigitalSignatureEx\n\n");
|
||||||
|
msi_dirent_free(dirent);
|
||||||
BIO_free_all(hash);
|
BIO_free_all(hash);
|
||||||
BIO_free_all(prehash);
|
BIO_free_all(prehash);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
msi_dirent_free(dirent);
|
||||||
BIO_gets(prehash, (char*)cexmdbuf, EVP_MAX_MD_SIZE);
|
BIO_gets(prehash, (char*)cexmdbuf, EVP_MAX_MD_SIZE);
|
||||||
BIO_free_all(prehash);
|
BIO_free_all(prehash);
|
||||||
BIO_write(hash, (char*)cexmdbuf, EVP_MD_size(md));
|
BIO_write(hash, (char*)cexmdbuf, EVP_MD_size(md));
|
||||||
#else
|
|
||||||
BIO_write(hash, (char *)exdata, EVP_MD_size(md));
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
if (!msi_handle_dir(infile, NULL, hash)) {
|
if (!msi_handle_dir(infile, NULL, hash)) {
|
||||||
printf("Failed to write a new output file\n\n");
|
printf("Failed to write a new output file\n\n");
|
||||||
@ -3216,7 +3349,6 @@ static int msi_verify_pkcs7(SIGNATURE *signature, GsfInfile *infile, unsigned ch
|
|||||||
printf("\n");
|
printf("\n");
|
||||||
|
|
||||||
if (exdata) {
|
if (exdata) {
|
||||||
#ifdef GSF_CAN_READ_MSI_METADATA
|
|
||||||
int exok;
|
int exok;
|
||||||
tohex(cexmdbuf, hexbuf, EVP_MD_size(md));
|
tohex(cexmdbuf, hexbuf, EVP_MD_size(md));
|
||||||
exok = !memcmp(exdata, cexmdbuf, MIN((size_t)EVP_MD_size(md), exlen));
|
exok = !memcmp(exdata, cexmdbuf, MIN((size_t)EVP_MD_size(md), exlen));
|
||||||
@ -3228,11 +3360,6 @@ static int msi_verify_pkcs7(SIGNATURE *signature, GsfInfile *infile, unsigned ch
|
|||||||
goto out;
|
goto out;
|
||||||
} else
|
} else
|
||||||
printf("\n");
|
printf("\n");
|
||||||
#else
|
|
||||||
tohex(exdata, hexbuf, MIN((size_t)EVP_MD_size(md), exlen));
|
|
||||||
printf("\nWarning: MsiDigitalSignatureEx found but not verified\n");
|
|
||||||
printf("Current MsiDigitalSignatureEx : %s\n\n", hexbuf);
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = verify_signature(signature, options);
|
ret = verify_signature(signature, options);
|
||||||
@ -3588,20 +3715,22 @@ static int msi_check_MsiDigitalSignatureEx(GsfInfile *ole, const EVP_MD *md)
|
|||||||
* ("metadata") hash.
|
* ("metadata") hash.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static int msi_calc_MsiDigitalSignatureEx(GsfInfile *ole, const EVP_MD *md,
|
static int msi_calc_MsiDigitalSignatureEx(GsfInfile *infile, const EVP_MD *md,
|
||||||
BIO *hash, GSF_PARAMS *gsfparams)
|
BIO *hash, GSF_PARAMS *gsfparams)
|
||||||
{
|
{
|
||||||
BIO *prehash;
|
MSOleDirent *dirent;
|
||||||
|
BIO *prehash= BIO_new(BIO_f_md());
|
||||||
prehash = BIO_new(BIO_f_md());
|
|
||||||
BIO_set_md(prehash, md);
|
BIO_set_md(prehash, md);
|
||||||
BIO_push(prehash, BIO_new(BIO_s_null()));
|
BIO_push(prehash, BIO_new(BIO_s_null()));
|
||||||
|
|
||||||
if (!msi_prehash(ole, NULL, prehash)) {
|
dirent = msi_dirent_new(GSF_INFILE_MSOLE(infile), 0, NULL);
|
||||||
|
if (!msi_prehash_dir(dirent, prehash)) {
|
||||||
printf("Unable to calculate MSI pre-hash ('metadata') hash\n");
|
printf("Unable to calculate MSI pre-hash ('metadata') hash\n");
|
||||||
|
msi_dirent_free(dirent);
|
||||||
BIO_free_all(prehash);
|
BIO_free_all(prehash);
|
||||||
return 0; /* FAILED */
|
return 0; /* FAILED */
|
||||||
}
|
}
|
||||||
|
msi_dirent_free(dirent);
|
||||||
gsfparams->p_msiex = OPENSSL_malloc(EVP_MAX_MD_SIZE);
|
gsfparams->p_msiex = OPENSSL_malloc(EVP_MAX_MD_SIZE);
|
||||||
gsfparams->len_msiex = BIO_gets(prehash, (char*)gsfparams->p_msiex, EVP_MAX_MD_SIZE);
|
gsfparams->len_msiex = BIO_gets(prehash, (char*)gsfparams->p_msiex, EVP_MAX_MD_SIZE);
|
||||||
BIO_write(hash, gsfparams->p_msiex, gsfparams->len_msiex);
|
BIO_write(hash, gsfparams->p_msiex, gsfparams->len_msiex);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user