|
|
|
/* Copyright (C) 2021 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __APP_LAYER_HTP_RANGE_H__
|
|
|
|
#define __APP_LAYER_HTP_RANGE_H__
|
|
|
|
|
|
|
|
#include "util-thash.h"
|
|
|
|
|
|
|
|
void HttpRangeContainersInit(void);
|
|
|
|
void HttpRangeContainersDestroy(void);
|
|
|
|
uint32_t HttpRangeContainersTimeoutHash(struct timeval *ts);
|
|
|
|
|
|
|
|
void *HttpRangeContainerUrlGet(const uint8_t *key, size_t keylen, struct timeval *ts);
|
|
|
|
|
|
|
|
// linked list of ranges : buffer with offset
|
|
|
|
typedef struct HttpRangeContainerBuffer {
|
|
|
|
/** red and black tree */
|
|
|
|
RB_ENTRY(HttpRangeContainerBuffer) rb;
|
|
|
|
/** allocated buffer */
|
|
|
|
uint8_t *buffer;
|
|
|
|
/** length of buffer */
|
|
|
|
uint64_t buflen;
|
|
|
|
/** the start of the range (offset relative to the absolute beginning of the file) */
|
|
|
|
uint64_t start;
|
|
|
|
/** offset of bytes written in buffer (relative to the start of the range) */
|
|
|
|
uint64_t offset;
|
|
|
|
} HttpRangeContainerBuffer;
|
|
|
|
|
|
|
|
int HttpRangeContainerBufferCompare(HttpRangeContainerBuffer *a, HttpRangeContainerBuffer *b);
|
|
|
|
|
|
|
|
RB_HEAD(HTTP_RANGES, HttpRangeContainerBuffer);
|
|
|
|
RB_PROTOTYPE(HTTP_RANGES, HttpRangeContainerBuffer, rb, HttpRangeContainerBufferCompare);
|
|
|
|
|
|
|
|
/** Item in hash table for a file in multiple ranges
|
|
|
|
* Thread-safety is ensured by the thread-safe hash table
|
|
|
|
* The number of use is increased for each flow opening a new HttpRangeContainerBlock
|
|
|
|
* until it closes this HttpRangeContainerBlock
|
|
|
|
*/
|
|
|
|
typedef struct HttpRangeContainerFile {
|
|
|
|
/** key for hashtable */
|
|
|
|
uint8_t *key;
|
|
|
|
/** key length */
|
|
|
|
uint32_t len;
|
|
|
|
/** expire time in epoch */
|
|
|
|
uint32_t expire;
|
|
|
|
/** pointer to hashtable data, for locking and use count */
|
|
|
|
THashData *hdata;
|
|
|
|
/** total epxected size of the file in ranges */
|
|
|
|
uint64_t totalsize;
|
|
|
|
/** file container, with only one file */
|
|
|
|
FileContainer *files;
|
|
|
|
/** red and black tree list of ranges which came out of order */
|
|
|
|
struct HTTP_RANGES fragment_tree;
|
|
|
|
/** file flags */
|
|
|
|
uint16_t flags;
|
|
|
|
/** wether a range file is currently appending */
|
|
|
|
bool appending;
|
|
|
|
/** error condition for this range. Its up to timeout handling to cleanup */
|
|
|
|
bool error;
|
|
|
|
} HttpRangeContainerFile;
|
|
|
|
|
|
|
|
/** A structure representing a single range request :
|
|
|
|
* either skipping, buffering, or appending
|
|
|
|
* As this belongs to a flow, appending data to it is ensured to be thread-safe
|
|
|
|
* Only one block per file has the pointer to the container
|
|
|
|
*/
|
|
|
|
typedef struct HttpRangeContainerBlock {
|
|
|
|
/** state where we skip content */
|
|
|
|
uint64_t toskip;
|
|
|
|
/** current out of order range to write into */
|
|
|
|
HttpRangeContainerBuffer *current;
|
|
|
|
/** pointer to the main file container, where to directly append data */
|
|
|
|
HttpRangeContainerFile *container;
|
|
|
|
} HttpRangeContainerBlock;
|
|
|
|
|
|
|
|
int HttpRangeProcessSkip(HttpRangeContainerBlock *c, const uint8_t *data, const uint32_t len);
|
|
|
|
int HttpRangeAppendData(HttpRangeContainerBlock *c, const uint8_t *data, uint32_t len);
|
|
|
|
File *HttpRangeClose(HttpRangeContainerBlock *c, uint16_t flags);
|
|
|
|
|
|
|
|
HttpRangeContainerBlock *HttpRangeOpenFile(HttpRangeContainerFile *c, uint64_t start, uint64_t end,
|
|
|
|
uint64_t total, const StreamingBufferConfig *sbcfg, const uint8_t *name, uint16_t name_len,
|
|
|
|
uint16_t flags, const uint8_t *data, uint32_t len);
|
|
|
|
|
|
|
|
void HttpRangeFreeBlock(HttpRangeContainerBlock *b);
|
|
|
|
|
|
|
|
#endif /* __APP_LAYER_HTP_RANGE_H__ */
|