2017-10-01 20:33:01 -07:00
|
|
|
/* ioapi_crypt.c -- IO base function header for compress/uncompress .zip
|
|
|
|
files using zlib + zip or unzip API
|
|
|
|
|
|
|
|
This version of ioapi is designed to access memory rather than files.
|
|
|
|
We do use a region of memory to put data in to and take it out of. We do
|
|
|
|
not have auto-extending buffers and do not inform anyone else that the
|
|
|
|
data has been written. It is really intended for accessing a zip archive
|
|
|
|
embedded in an application such that I can write an installer with no
|
|
|
|
external files. Creation of archives has not been attempted, although
|
|
|
|
parts of the framework are present.
|
|
|
|
|
|
|
|
Based on Unzip ioapi.c version 0.22, May 19th, 2003
|
|
|
|
|
|
|
|
Copyright (C) 2012-2017 Nathan Moinvaziri
|
|
|
|
https://github.com/nmoinvaz/minizip
|
|
|
|
Copyright (C) 2003 Justin Fletcher
|
|
|
|
Copyright (C) 1998-2003 Gilles Vollant
|
|
|
|
http://www.winimage.com/zLibDll/minizip.html
|
|
|
|
|
|
|
|
This file is under the same license as the Unzip tool it is distributed
|
|
|
|
with.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "zlib.h"
|
|
|
|
#include "ioapi.h"
|
|
|
|
|
|
|
|
#include "ioapi_crypt.h"
|
|
|
|
|
|
|
|
#define RAND_HEAD_LEN 12
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
typedef struct mz_stream_crypt_s {
|
|
|
|
mz_stream stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
uint32_t keys[3]; /* keys defining the pseudo-random sequence */
|
|
|
|
const z_crc_t *crc_32_tab;
|
|
|
|
int16_t initialized;
|
|
|
|
int16_t error;
|
|
|
|
uint8_t verify1;
|
|
|
|
uint8_t verify2;
|
|
|
|
char *password;
|
|
|
|
uint64_t total_in;
|
|
|
|
uint64_t total_out;
|
2017-10-01 21:43:24 -07:00
|
|
|
} mz_stream_crypt;
|
2017-10-01 20:33:01 -07:00
|
|
|
|
|
|
|
#define zdecode(keys,crc_32_tab,c) \
|
2017-10-01 21:43:24 -07:00
|
|
|
(mz_stream_crypt_update_keys(keys,crc_32_tab, c ^= mz_stream_crypt_decrypt_byte(keys)))
|
2017-10-01 20:33:01 -07:00
|
|
|
|
|
|
|
#define zencode(keys,crc_32_tab,c,t) \
|
2017-10-01 21:43:24 -07:00
|
|
|
(t = mz_stream_crypt_decrypt_byte(keys), mz_stream_crypt_update_keys(keys,crc_32_tab,c), t^(c))
|
2017-10-01 20:33:01 -07:00
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
uint8_t mz_stream_crypt_decrypt_byte(uint32_t *keys)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
|
|
|
unsigned temp; /* POTENTIAL BUG: temp*(temp^1) may overflow in an
|
|
|
|
* unpredictable manner on 16-bit systems; not a problem
|
|
|
|
* with any known compiler so far, though */
|
|
|
|
|
|
|
|
temp = ((uint32_t)(*(keys+2)) & 0xffff) | 2;
|
|
|
|
return (uint8_t)(((temp * (temp ^ 1)) >> 8) & 0xff);
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
uint8_t mz_stream_crypt_update_keys(uint32_t *keys, const z_crc_t *crc_32_tab, int32_t c)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
|
|
|
#define CRC32(c, b) ((*(crc_32_tab+(((uint32_t)(c) ^ (b)) & 0xff))) ^ ((c) >> 8))
|
|
|
|
|
|
|
|
(*(keys+0)) = (uint32_t)CRC32((*(keys+0)), c);
|
|
|
|
(*(keys+1)) += (*(keys+0)) & 0xff;
|
|
|
|
(*(keys+1)) = (*(keys+1)) * 134775813L + 1;
|
|
|
|
{
|
|
|
|
register int32_t keyshift = (int32_t)((*(keys + 1)) >> 24);
|
|
|
|
(*(keys+2)) = (uint32_t)CRC32((*(keys+2)), keyshift);
|
|
|
|
}
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
void mz_stream_crypt_init_keys(const char *password, uint32_t *keys, const z_crc_t *crc_32_tab)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
|
|
|
*(keys+0) = 305419896L;
|
|
|
|
*(keys+1) = 591751049L;
|
|
|
|
*(keys+2) = 878082192L;
|
|
|
|
while (*password != 0)
|
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt_update_keys(keys, crc_32_tab, *password);
|
2017-10-01 20:33:01 -07:00
|
|
|
password += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
int32_t ZCALLBACK mz_stream_crypt_open(voidpf stream, const char *filename, int mode)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
uint16_t t = 0;
|
|
|
|
int16_t i = 0;
|
|
|
|
uint8_t header[RAND_HEAD_LEN];
|
|
|
|
uint8_t verify1 = 0;
|
|
|
|
uint8_t verify2 = 0;
|
|
|
|
|
|
|
|
crypt->total_in = 0;
|
|
|
|
crypt->total_out = 0;
|
|
|
|
crypt->initialized = 0;
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
if (mz_stream_is_open(crypt->stream.base) == MZ_STREAM_ERR)
|
|
|
|
return MZ_STREAM_ERR;
|
2017-10-01 20:33:01 -07:00
|
|
|
if (crypt->password == NULL)
|
2017-10-01 21:43:24 -07:00
|
|
|
return MZ_STREAM_ERR;
|
2017-10-01 20:33:01 -07:00
|
|
|
|
|
|
|
crypt->crc_32_tab = get_crc_table();
|
|
|
|
if (crypt->crc_32_tab == NULL)
|
2017-10-01 21:43:24 -07:00
|
|
|
return MZ_STREAM_ERR;
|
2017-10-01 20:33:01 -07:00
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt_init_keys(crypt->password, crypt->keys, crypt->crc_32_tab);
|
2017-10-01 20:33:01 -07:00
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
if (mode & MZ_STREAM_MODE_WRITE)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
|
|
|
// First generate RAND_HEAD_LEN - 2 random bytes.
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_os_rand(header, RAND_HEAD_LEN - 2);
|
2017-10-01 20:33:01 -07:00
|
|
|
|
|
|
|
// Encrypt random header (last two bytes is high word of crc)
|
|
|
|
for (i = 0; i < RAND_HEAD_LEN - 2; i++)
|
|
|
|
header[i] = (uint8_t)zencode(crypt->keys, crypt->crc_32_tab, header[i], t);
|
|
|
|
|
|
|
|
header[i++] = (uint8_t)zencode(crypt->keys, crypt->crc_32_tab, crypt->verify1, t);
|
|
|
|
header[i++] = (uint8_t)zencode(crypt->keys, crypt->crc_32_tab, crypt->verify2, t);
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
if (mz_stream_write(crypt->stream.base, header, RAND_HEAD_LEN) != RAND_HEAD_LEN)
|
|
|
|
return MZ_STREAM_ERR;
|
2017-10-01 20:33:01 -07:00
|
|
|
|
|
|
|
crypt->total_out += RAND_HEAD_LEN;
|
|
|
|
}
|
2017-10-01 21:43:24 -07:00
|
|
|
else if (mode & MZ_STREAM_MODE_READ)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
if (mz_stream_read(crypt->stream.base, header, RAND_HEAD_LEN) != RAND_HEAD_LEN)
|
|
|
|
return MZ_STREAM_ERR;
|
2017-10-01 20:33:01 -07:00
|
|
|
|
|
|
|
for (i = 0; i < RAND_HEAD_LEN - 2; i++)
|
|
|
|
header[i] = (uint8_t)zdecode(crypt->keys, crypt->crc_32_tab, header[i]);
|
|
|
|
|
|
|
|
crypt->verify1 = (uint8_t)zdecode(crypt->keys, crypt->crc_32_tab, header[i++]);
|
|
|
|
crypt->verify2 = (uint8_t)zdecode(crypt->keys, crypt->crc_32_tab, header[i++]);
|
|
|
|
|
|
|
|
crypt->total_in += RAND_HEAD_LEN;
|
|
|
|
}
|
|
|
|
|
|
|
|
crypt->initialized = 1;
|
2017-10-01 21:43:24 -07:00
|
|
|
return MZ_STREAM_OK;
|
2017-10-01 20:33:01 -07:00
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
int32_t ZCALLBACK mz_stream_crypt_is_open(voidpf stream)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
if (crypt->initialized == 0)
|
2017-10-01 21:43:24 -07:00
|
|
|
return MZ_STREAM_ERR;
|
|
|
|
return MZ_STREAM_OK;
|
2017-10-01 20:33:01 -07:00
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
int32_t ZCALLBACK mz_stream_crypt_read(voidpf stream, void *buf, uint32_t size)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
uint8_t *buf_ptr = (uint8_t *)buf;
|
|
|
|
uint32_t read = 0;
|
|
|
|
uint32_t i = 0;
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
read = mz_stream_read(crypt->stream.base, buf, size);
|
2017-10-01 20:33:01 -07:00
|
|
|
for (i = 0; i < read; i++)
|
|
|
|
buf_ptr[i] = (uint8_t)zdecode(crypt->keys, crypt->crc_32_tab, buf_ptr[i]);
|
|
|
|
crypt->total_in += read;
|
|
|
|
return read;
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
int32_t ZCALLBACK mz_stream_crypt_write(voidpf stream, const void *buf, uint32_t size)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
uint8_t *buf_ptr = (uint8_t *)buf;
|
|
|
|
uint32_t written = 0;
|
|
|
|
uint32_t i = 0;
|
|
|
|
uint16_t t = 0;
|
|
|
|
for (i = 0; i < size; i++)
|
|
|
|
buf_ptr[i] = (uint8_t)zencode(crypt->keys, crypt->crc_32_tab, buf_ptr[i], t);
|
2017-10-01 21:43:24 -07:00
|
|
|
written = mz_stream_write(crypt->stream.base, buf, size);
|
2017-10-01 20:33:01 -07:00
|
|
|
if (written > 0)
|
|
|
|
crypt->total_out += written;
|
|
|
|
if (written != size)
|
2017-10-01 21:43:24 -07:00
|
|
|
return MZ_STREAM_ERR;
|
2017-10-01 20:33:01 -07:00
|
|
|
return written;
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
int64_t ZCALLBACK mz_stream_crypt_tell(voidpf stream)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
|
|
|
return mz_stream_tell(crypt->stream.base);
|
2017-10-01 20:33:01 -07:00
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
int32_t ZCALLBACK mz_stream_crypt_seek(voidpf stream, uint64_t offset, int origin)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
|
|
|
return mz_stream_seek(crypt->stream.base, offset, origin);
|
2017-10-01 20:33:01 -07:00
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
int32_t ZCALLBACK mz_stream_crypt_close(voidpf stream)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
crypt->initialized = 0;
|
2017-10-01 21:43:24 -07:00
|
|
|
return MZ_STREAM_OK;
|
2017-10-01 20:33:01 -07:00
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
int32_t ZCALLBACK mz_stream_crypt_error(voidpf stream)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
return crypt->error;
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
void mz_stream_crypt_set_password(voidpf stream, char *password)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
crypt->password = password;
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
void mz_stream_crypt_set_verify(voidpf stream, uint8_t verify1, uint8_t verify2)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
crypt->verify1 = verify1;
|
|
|
|
crypt->verify2 = verify2;
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
void mz_stream_crypt_get_verify(voidpf stream, uint8_t *verify1, uint8_t *verify2)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = (mz_stream_crypt *)stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
*verify1 = crypt->verify1;
|
|
|
|
*verify2 = crypt->verify2;
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
voidpf mz_stream_crypt_create(voidpf *stream)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = NULL;
|
2017-10-01 20:33:01 -07:00
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
crypt = (mz_stream_crypt *)malloc(sizeof(mz_stream_crypt));
|
|
|
|
if (crypt != NULL)
|
|
|
|
{
|
|
|
|
memset(crypt, 0, sizeof(mz_stream_crypt));
|
|
|
|
|
|
|
|
crypt->stream.open = mz_stream_crypt_open;
|
|
|
|
crypt->stream.is_open = mz_stream_crypt_is_open;
|
|
|
|
crypt->stream.read = mz_stream_crypt_read;
|
|
|
|
crypt->stream.write = mz_stream_crypt_write;
|
|
|
|
crypt->stream.tell = mz_stream_crypt_tell;
|
|
|
|
crypt->stream.seek = mz_stream_crypt_seek;
|
|
|
|
crypt->stream.close = mz_stream_crypt_close;
|
|
|
|
crypt->stream.error = mz_stream_crypt_error;
|
|
|
|
crypt->stream.create = mz_stream_crypt_create;
|
|
|
|
crypt->stream.delete = mz_stream_crypt_delete;
|
|
|
|
}
|
2017-10-01 20:33:01 -07:00
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
if (stream != NULL)
|
|
|
|
*stream = crypt;
|
2017-10-01 20:33:01 -07:00
|
|
|
return (voidpf)crypt;
|
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
void mz_stream_crypt_delete(voidpf *stream)
|
2017-10-01 20:33:01 -07:00
|
|
|
{
|
2017-10-01 21:43:24 -07:00
|
|
|
mz_stream_crypt *crypt = NULL;
|
|
|
|
if (stream == NULL)
|
|
|
|
return;
|
|
|
|
crypt = (mz_stream_crypt *)*stream;
|
2017-10-01 20:33:01 -07:00
|
|
|
if (crypt != NULL)
|
|
|
|
free(crypt);
|
|
|
|
}
|