uri/tinfgzip: Minor fixes and layering improvement

This commit is contained in:
mintsuki 2023-02-16 02:05:11 +01:00
parent ccb0d0f738
commit 24af4c116e
3 changed files with 34 additions and 24 deletions

View File

@ -24,6 +24,7 @@
* distribution.
*/
#include <stdint.h>
#include <stb/stb_image.h>
typedef enum {
@ -34,29 +35,27 @@ typedef enum {
FCOMMENT = 16
} tinf_gzip_flag;
int tinf_gzip_uncompress(void *dest, unsigned int limit,
const void *source, unsigned int sourceLen) {
const unsigned char *src = (const unsigned char *) source;
unsigned char *dst = (unsigned char *) dest;
const unsigned char *start;
void *tinf_gzip_uncompress(const void *source, uint64_t sourceLen, uint64_t *outsize) {
const uint8_t *src = (const uint8_t *) source;
const uint8_t *start;
int res;
unsigned char flg;
uint8_t flg;
/* -- Check header -- */
/* Check room for at least 10 byte header and 8 byte trailer */
if (sourceLen < 18) {
return -1;
return NULL;
}
/* Check id bytes */
if (src[0] != 0x1F || src[1] != 0x8B) {
return -1;
return NULL;
}
/* Check method is deflate */
if (src[2] != 8) {
return -1;
return NULL;
}
/* Get flag byte */
@ -64,7 +63,7 @@ int tinf_gzip_uncompress(void *dest, unsigned int limit,
/* Check that reserved bits are zero */
if (flg & 0xE0) {
return -1;
return NULL;
}
/* -- Find start of compressed data -- */
@ -74,10 +73,10 @@ int tinf_gzip_uncompress(void *dest, unsigned int limit,
/* Skip extra data if present */
if (flg & FEXTRA) {
unsigned int xlen = *start;
uint64_t xlen = *((uint16_t *)start);
if (xlen > sourceLen - 12) {
return -1;
return NULL;
}
start += xlen + 2;
@ -86,8 +85,8 @@ int tinf_gzip_uncompress(void *dest, unsigned int limit,
/* Skip file name if present */
if (flg & FNAME) {
do {
if (((unsigned int)(start - src)) >= sourceLen) {
return -1;
if (((uint64_t)(start - src)) >= sourceLen) {
return NULL;
}
} while (*start++);
}
@ -95,8 +94,8 @@ int tinf_gzip_uncompress(void *dest, unsigned int limit,
/* Skip file comment if present */
if (flg & FCOMMENT) {
do {
if (((unsigned int)(start - src)) >= sourceLen) {
return -1;
if (((uint64_t)(start - src)) >= sourceLen) {
return NULL;
}
} while (*start++);
}
@ -105,13 +104,25 @@ int tinf_gzip_uncompress(void *dest, unsigned int limit,
start += 2;
}
/* -- Get decompressed length -- */
uint32_t dlen = *((uint32_t *)&src[sourceLen - 4]);
/* -- Decompress data -- */
if ((src + sourceLen) - start < 8) {
return -1;
return NULL;
}
res = stbi_zlib_decode_noheader_buffer((char *)dst, limit, (const char *)start, (src + sourceLen) - start - 8);
void *buf = ext_mem_alloc(dlen);
return res == -1 ? res : 0;
res = stbi_zlib_decode_noheader_buffer(buf, dlen, (const char *)start, (src + sourceLen) - start - 8);
if (res == -1) {
pmm_free(buf, dlen);
return NULL;
}
*outsize = (uint64_t)dlen;
return buf;
}

View File

@ -1,7 +1,8 @@
#ifndef __COMPRESS__TINFGZIP_H__
#define __COMPRESS__TINFGZIP_H__
int tinf_gzip_uncompress(void *dest, unsigned int limit,
const void *source, unsigned int sourceLen);
#include <stdint.h>
void *tinf_gzip_uncompress(const void *source, uint64_t sourceLen, uint64_t *outsize);
#endif

View File

@ -272,10 +272,8 @@ struct file_handle *uri_open(char *uri) {
if (compressed && ret != NULL) {
struct file_handle *compressed_fd = ext_mem_alloc(sizeof(struct file_handle));
fread(ret, &compressed_fd->size, ret->size - 4, sizeof(uint32_t));
compressed_fd->fd = ext_mem_alloc(compressed_fd->size);
void *src = freadall(ret, MEMMAP_BOOTLOADER_RECLAIMABLE);
if (tinf_gzip_uncompress(compressed_fd->fd, compressed_fd->size, src, ret->size)) {
if ((compressed_fd->fd = tinf_gzip_uncompress(src, ret->size, &compressed_fd->size)) == NULL) {
panic(true, "tinf error");
}
compressed_fd->vol = ret->vol;