You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
suricata/src/util-file-decompression.c

202 lines
6.8 KiB
C

/* Copyright (C) 2017 Open Information Security Foundation
*
* You can copy, redistribute or modify this Program under the terms of
* the GNU General Public License version 2 as published by the Free
* Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* version 2 along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301, USA.
*/
/** \file
*
* \author Giuseppe Longo <giuseppe@glongo.it>
*
* \brief Decompress files transfered via HTTP corresponding to file_data
* keyword.
*
*/
#include "suricata-common.h"
#include "suricata.h"
#include "detect-engine.h"
#include "app-layer-htp.h"
#include "util-file-decompression.h"
#include "util-file-swf-decompression.h"
#include "util-misc.h"
#include "util-print.h"
#define SWF_ZLIB_MIN_VERSION 0x06
#define SWF_LZMA_MIN_VERSION 0x0D
int FileIsSwfFile(const uint8_t *buffer, uint32_t buffer_len)
{
if (buffer_len >= 3 && buffer[1] == 'W' && buffer[2] == 'S') {
if (buffer[0] == 'F')
return FILE_SWF_NO_COMPRESSION;
else if (buffer[0] == 'C')
return FILE_SWF_ZLIB_COMPRESSION;
else if (buffer[0] == 'Z')
return FILE_SWF_LZMA_COMPRESSION;
else
return FILE_IS_NOT_SWF;
}
return FILE_IS_NOT_SWF;
}
/**
* \brief This function decompresses a buffer with zlib/lzma algorithm
*
* \param buffer compressed buffer
* \param buffer_len compressed buffer length
* \param decompressed_buffer buffer that store decompressed data
* \param decompressed_buffer_len decompressesd data length
* \param swf_type decompression algorithm to use
* \param decompress_depth how much decompressed data we want to store
* \param compress_depth how much compressed data we want to decompress
*
* \retval 1 if decompression works
* \retval 0 an error occured, and event set
*/
int FileSwfDecompression(const uint8_t *buffer, uint32_t buffer_len,
DetectEngineThreadCtx *det_ctx,
InspectionBuffer *out_buffer,
int swf_type,
uint32_t decompress_depth,
uint32_t compress_depth)
{
int r = 0;
int compression_type = FileIsSwfFile(buffer, buffer_len);
if (compression_type == FILE_SWF_NO_COMPRESSION) {
return 0;
}
uint32_t offset = 0;
if (compression_type == FILE_SWF_ZLIB_COMPRESSION) {
/* compressed data start from the 4th bytes */
offset = 8;
} else if (compression_type == FILE_SWF_LZMA_COMPRESSION) {
/* compressed data start from the 17th bytes */
offset = 17;
}
if (buffer_len <= offset) {
DetectEngineSetEvent(det_ctx, FILE_DECODER_EVENT_INVALID_SWF_LENGTH);
return 0;
}
uint32_t compressed_data_len = 0;
if (compress_depth > 0 && compress_depth <= buffer_len - offset) {
compressed_data_len = compress_depth;
} else {
compressed_data_len = buffer_len - offset;
}
/* get swf version */
uint8_t swf_version = FileGetSwfVersion(buffer, buffer_len);
if (compression_type == FILE_SWF_ZLIB_COMPRESSION &&
swf_version < SWF_ZLIB_MIN_VERSION)
{
DetectEngineSetEvent(det_ctx, FILE_DECODER_EVENT_INVALID_SWF_VERSION);
return 0;
}
if (compression_type == FILE_SWF_LZMA_COMPRESSION &&
swf_version < SWF_LZMA_MIN_VERSION)
{
DetectEngineSetEvent(det_ctx, FILE_DECODER_EVENT_INVALID_SWF_VERSION);
return 0;
}
/* get flash decompressed file length */
uint32_t decompressed_swf_len = FileGetSwfDecompressedLen(buffer, buffer_len);
if (decompressed_swf_len == 0) {
decompressed_swf_len = MIN_SWF_LEN;
}
/* if decompress_depth is 0, keep the flash file length */
uint32_t decompressed_data_len = (decompress_depth == 0) ? decompressed_swf_len : decompress_depth;
decompressed_data_len += 8;
/* make sure the inspection buffer has enough space */
InspectionBufferCheckAndExpand(out_buffer, decompressed_data_len);
if (out_buffer->size < decompressed_data_len) {
DetectEngineSetEvent(det_ctx, FILE_DECODER_EVENT_NO_MEM);
return 0;
}
out_buffer->len = decompressed_data_len;
/*
* FWS format
* | 4 bytes | 4 bytes | n bytes |
* | 'FWS' + version | script len | data |
*/
out_buffer->buf[0] = 'F';
out_buffer->buf[1] = 'W';
out_buffer->buf[2] = 'S';
out_buffer->buf[3] = swf_version;
memcpy(out_buffer->buf + 4, &decompressed_swf_len, 4);
memset(out_buffer->buf + 8, 0, decompressed_data_len - 8);
if ((swf_type == HTTP_SWF_COMPRESSION_ZLIB || swf_type == HTTP_SWF_COMPRESSION_BOTH) &&
compression_type == FILE_SWF_ZLIB_COMPRESSION)
{
/* the first 8 bytes represents the fws header, see 'FWS format' above.
* data will start from 8th bytes
*/
r = FileSwfZlibDecompression(det_ctx,
(uint8_t *)buffer + offset, compressed_data_len,
out_buffer->buf + 8, out_buffer->len - 8);
if (r == 0)
goto error;
} else if ((swf_type == HTTP_SWF_COMPRESSION_LZMA || swf_type == HTTP_SWF_COMPRESSION_BOTH) &&
compression_type == FILE_SWF_LZMA_COMPRESSION)
{
/* we need to setup the lzma header */
/*
* | 5 bytes | 8 bytes | n bytes |
* | LZMA properties | Uncompressed length | Compressed data |
*/
compressed_data_len += 13;
uint8_t compressed_data[compressed_data_len];
/* put lzma properties */
memcpy(compressed_data, buffer + 12, 5);
/* put lzma end marker */
memset(compressed_data + 5, 0xFF, 8);
/* put compressed data */
memcpy(compressed_data + 13, buffer + offset, compressed_data_len - 13);
/* the first 8 bytes represents the fws header, see 'FWS format' above.
* data will start from 8th bytes
*/
r = FileSwfLzmaDecompression(det_ctx,
compressed_data, compressed_data_len,
out_buffer->buf + 8, out_buffer->len - 8);
if (r == 0)
goto error;
} else {
goto error;
}
/* all went well so switch the buffer's inspect pointer/size
* to use the new data. */
out_buffer->inspect = out_buffer->buf;
out_buffer->inspect_len = out_buffer->len;
return 1;
error:
return 0;
}