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/app-layer-frames.c

786 lines
26 KiB
C

/* Copyright (C) 2007-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 frameeived 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 Victor Julien <victor@inliniac.net>
*
*/
#include "suricata-common.h"
#include "debug.h"
#include "util-print.h"
#include "stream-tcp.h"
#include "app-layer-frames.h"
static void FrameDebug(const char *prefix, const Frames *frames, const Frame *frame)
{
#ifdef DEBUG
const char *type_name =
frames ? AppLayerParserGetFrameNameById(frames->ipproto, frames->alproto, frame->type)
: "<unknown>";
SCLogDebug("[%s] %p: frame: %p type %u/%s id %" PRIi64 " flags %02x rel_offset:%" PRIi64
", len:%" PRIi64 ", events:%u %u/%u/%u/%u",
prefix, frames, frame, frame->type, type_name, frame->id, frame->flags,
frame->rel_offset, frame->len, frame->event_cnt, frame->events[0], frame->events[1],
frame->events[2], frame->events[3]);
#endif
}
Frame *FrameGetById(Frames *frames, const int64_t id)
{
for (uint16_t i = 0; i < frames->cnt; i++) {
if (i < FRAMES_STATIC_CNT) {
Frame *frame = &frames->sframes[i];
if (frame->id == id)
return frame;
} else {
const uint16_t o = i - FRAMES_STATIC_CNT;
Frame *frame = &frames->dframes[o];
if (frame->id == id)
return frame;
}
}
return NULL;
}
Frame *FrameGetByIndex(Frames *frames, const uint32_t idx)
{
if (idx >= frames->cnt)
return NULL;
if (idx < FRAMES_STATIC_CNT) {
Frame *frame = &frames->sframes[idx];
FrameDebug("get_by_idx(s)", frames, frame);
return frame;
} else {
const uint32_t o = idx - FRAMES_STATIC_CNT;
Frame *frame = &frames->dframes[o];
FrameDebug("get_by_idx(d)", frames, frame);
return frame;
}
}
// TODO review rel_offset logic. App-layer passes STREAM_APP_PROGRESS as
// offset, but I think we're using rel_offset relative to BASE_PROGRESS
// here which changes only on slide.
static Frame *FrameNew(Frames *frames, int64_t rel_offset, int64_t len)
{
BUG_ON(frames == NULL);
if (frames->cnt < FRAMES_STATIC_CNT) {
Frame *frame = &frames->sframes[frames->cnt];
frames->sframes[frames->cnt].rel_offset = rel_offset;
frames->sframes[frames->cnt].len = len;
frames->sframes[frames->cnt].id = ++frames->base_id;
frames->cnt++;
return frame;
} else if (frames->dframes == NULL) {
BUG_ON(frames->dyn_size != 0);
BUG_ON(frames->cnt != FRAMES_STATIC_CNT);
frames->dframes = SCCalloc(8, sizeof(Frame));
if (frames->dframes == NULL) {
return NULL;
}
frames->cnt++;
BUG_ON(frames->cnt != FRAMES_STATIC_CNT + 1);
frames->dyn_size = 8;
frames->dframes[0].rel_offset = rel_offset;
frames->dframes[0].len = len;
frames->dframes[0].id = ++frames->base_id;
return &frames->dframes[0];
} else {
BUG_ON(frames->cnt < FRAMES_STATIC_CNT);
/* need to handle dynamic storage of frames now */
const uint16_t dyn_cnt = frames->cnt - FRAMES_STATIC_CNT;
if (dyn_cnt < frames->dyn_size) {
BUG_ON(frames->dframes == NULL);
// fall through
} else {
if (frames->dyn_size == 256) {
SCLogDebug("limit reached! 256 dynamic frames already");
// limit reached
// TODO figure out if this should lead to an event of sorts
return NULL;
}
/* realloc time */
uint16_t new_dyn_size = frames->dyn_size * 2;
uint32_t new_alloc_size = new_dyn_size * sizeof(Frame);
void *ptr = SCRealloc(frames->dframes, new_alloc_size);
if (ptr == NULL) {
return NULL;
}
memset((uint8_t *)ptr + (frames->dyn_size * sizeof(Frame)), 0x00,
(frames->dyn_size * sizeof(Frame)));
frames->dframes = ptr;
frames->dyn_size = new_dyn_size;
}
frames->cnt++;
frames->dframes[dyn_cnt].rel_offset = rel_offset;
frames->dframes[dyn_cnt].len = len;
frames->dframes[dyn_cnt].id = ++frames->base_id;
return &frames->dframes[dyn_cnt];
}
}
static void FrameClean(Frame *frame)
{
memset(frame, 0, sizeof(*frame));
}
static void FrameCopy(Frame *dst, Frame *src)
{
memcpy(dst, src, sizeof(*dst));
}
static void AppLayerFrameDumpForFrames(const char *prefix, const Frames *frames)
{
SCLogDebug("prefix: %s", prefix);
for (uint16_t i = 0; i < frames->cnt; i++) {
if (i < FRAMES_STATIC_CNT) {
const Frame *frame = &frames->sframes[i];
FrameDebug(prefix, frames, frame);
} else {
const uint16_t o = i - FRAMES_STATIC_CNT;
const Frame *frame = &frames->dframes[o];
FrameDebug(prefix, frames, frame);
}
}
SCLogDebug("prefix: %s", prefix);
}
static inline uint64_t FrameLeftEdge(
const TcpStream *stream, const Frame *frame, const int64_t base_offset)
{
const int64_t app_progress = STREAM_APP_PROGRESS(stream);
BUG_ON(base_offset > app_progress);
const int64_t frame_offset = base_offset + frame->rel_offset;
const int64_t frame_data = app_progress - frame_offset;
SCLogDebug("base_offset %" PRIi64 ", app_progress %" PRIi64, base_offset, app_progress);
SCLogDebug("frame_offset %" PRIi64 ", frame_data %" PRIi64 ", frame->len %" PRIi64,
frame_offset, frame_data, frame->len);
BUG_ON(frame_offset < 0);
BUG_ON(frame_offset > app_progress);
/* length unknown, make sure to have at least 2500 */
if (frame->len < 0) {
if (frame_data <= 2500) {
SCLogDebug("got <= 2500 bytes (%" PRIu64 "), returning offset %" PRIu64, frame_data,
frame_offset);
return frame_offset;
} else {
SCLogDebug("got > 2500 bytes (%" PRIu64 "), returning offset %" PRIu64, frame_data,
(frame_offset + (frame_data - 2500)));
return frame_offset + (frame_data - 2500);
}
/* length specified */
} else {
/* have all data for the frame, we can skip it */
if (frame->len <= frame_data) {
uint64_t x = frame_offset + frame_data;
SCLogDebug("x %" PRIu64, x);
return x;
/*
[ stream <frame_data> ]
[ frame .......]
*/
} else if (frame_data < 2500) {
uint64_t x = frame_offset;
SCLogDebug("x %" PRIu64, x);
return x;
} else {
uint64_t x = frame_offset + (frame_data - 2500);
SCLogDebug("x %" PRIu64, x);
return x;
}
}
}
/** Stream buffer slides forward, we need to update and age out
* frame offsets/frames. Aging out means we move existing frames
* into the slots we'd free up.
*
* Start:
*
* [ stream ]
* [ frame ...........]
* rel_offset: 2
* len: 19
*
* Slide:
* [ stream ]
* [ frame .... .]
* rel_offset: -10
* len: 19
*
* Slide:
* [ stream ]
* [ frame ........... ]
* rel_offset: -16
* len: 19
*/
static int FrameSlide(const char *ds, Frames *frames, const TcpStream *stream, const uint32_t slide)
{
SCLogDebug("start: left edge %" PRIu64 ", left_edge_rel %u, stream base %" PRIu64
", next %" PRIu64,
(uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream), frames->left_edge_rel,
STREAM_BASE_OFFSET(stream), STREAM_BASE_OFFSET(stream) + slide);
BUG_ON(frames == NULL);
SCLogDebug("%s frames %p: sliding %u bytes", ds, frames, slide);
uint64_t le = STREAM_APP_PROGRESS(stream);
const uint64_t next_base = STREAM_BASE_OFFSET(stream) + slide;
const uint16_t start = frames->cnt;
uint16_t removed = 0;
uint16_t x = 0;
for (uint16_t i = 0; i < frames->cnt; i++) {
if (i < FRAMES_STATIC_CNT) {
Frame *frame = &frames->sframes[i];
FrameDebug("slide(s)", frames, frame);
if (frame->len >= 0 &&
frame->rel_offset + frame->len <= (int64_t)slide) { // TODO check seems off
// remove by not incrementing 'x'
SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
FrameClean(frame);
removed++;
} else {
Frame *nframe = &frames->sframes[x];
FrameCopy(nframe, frame);
nframe->rel_offset -= slide; /* turns negative if start if before window */
if (frame != nframe) {
FrameClean(frame);
}
le = MIN(le, FrameLeftEdge(stream, nframe, next_base));
x++;
}
} else {
const uint16_t o = i - FRAMES_STATIC_CNT;
Frame *frame = &frames->dframes[o];
FrameDebug("slide(d)", frames, frame);
if (frame->len >= 0 && frame->rel_offset + frame->len <= (int64_t)slide) {
// remove by not incrementing 'x'
SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
FrameClean(frame);
removed++;
} else {
Frame *nframe;
if (x >= FRAMES_STATIC_CNT) {
nframe = &frames->dframes[x - FRAMES_STATIC_CNT];
} else {
nframe = &frames->sframes[x];
}
FrameCopy(nframe, frame);
nframe->rel_offset -= slide; /* turns negative if start is before window */
if (frame != nframe) {
FrameClean(frame);
}
le = MIN(le, FrameLeftEdge(stream, nframe, next_base));
x++;
}
}
}
frames->cnt = x;
uint64_t o = STREAM_BASE_OFFSET(stream) + slide;
frames->left_edge_rel = le - (STREAM_BASE_OFFSET(stream) + slide);
#ifdef DEBUG
SCLogDebug("end: left edge %" PRIu64 ", left_edge_rel %u, stream base %" PRIu64
" (+slide), cnt %u, removed %u, start %u",
(uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream) + slide,
frames->left_edge_rel, STREAM_BASE_OFFSET(stream) + slide, frames->cnt, removed, start);
char pf[32] = "";
snprintf(pf, sizeof(pf), "%s:post_slide", ds);
AppLayerFrameDumpForFrames(pf, frames);
#endif
BUG_ON(o > le);
BUG_ON(x != start - removed);
return 0;
}
void AppLayerFramesSlide(Flow *f, const uint32_t slide, const uint8_t direction)
{
FramesContainer *frames_container = AppLayerFramesGetContainer(f);
if (frames_container == NULL)
return;
Frames *frames;
TcpSession *ssn = f->protoctx;
TcpStream *stream;
if (direction == STREAM_TOSERVER) {
stream = &ssn->client;
frames = &frames_container->toserver;
FrameSlide("toserver", frames, stream, slide);
} else {
stream = &ssn->server;
frames = &frames_container->toclient;
FrameSlide("toclient", frames, stream, slide);
}
}
static void FrameFreeSingleFrame(Frames *frames, Frame *r)
{
FrameDebug("free", frames, r);
FrameClean(r);
}
static void FramesClear(Frames *frames)
{
BUG_ON(frames == NULL);
SCLogDebug("frames %u", frames->cnt);
for (uint16_t i = 0; i < frames->cnt; i++) {
if (i < FRAMES_STATIC_CNT) {
Frame *r = &frames->sframes[i];
SCLogDebug("removing frame %p", r);
FrameFreeSingleFrame(frames, r);
} else {
const uint16_t o = i - FRAMES_STATIC_CNT;
Frame *r = &frames->dframes[o];
SCLogDebug("removing frame %p", r);
FrameFreeSingleFrame(frames, r);
}
}
frames->cnt = 0;
}
void FramesFree(Frames *frames)
{
BUG_ON(frames == NULL);
FramesClear(frames);
SCFree(frames->dframes);
frames->dframes = NULL;
}
/** \brief create new frame using a pointer to start of the frame
*/
Frame *AppLayerFrameNewByPointer(Flow *f, const StreamSlice *stream_slice,
const uint8_t *frame_start, const int64_t len, int dir, uint8_t frame_type)
{
SCLogDebug("stream_slice offset %" PRIu64, stream_slice->offset);
SCLogDebug("frame_start %p stream_slice->input %p", frame_start, stream_slice->input);
/* workarounds for many (unit|fuzz)tests not handling TCP data properly */
#if defined(UNITTESTS) || defined(FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION)
if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
return NULL;
if (frame_start < stream_slice->input ||
frame_start >= stream_slice->input + stream_slice->input_len)
return NULL;
#endif
BUG_ON(frame_start < stream_slice->input);
BUG_ON(stream_slice->input == NULL);
BUG_ON(f->proto == IPPROTO_TCP && f->protoctx == NULL);
ptrdiff_t ptr_offset = frame_start - stream_slice->input;
#ifdef DEBUG
uint64_t offset = ptr_offset + stream_slice->offset;
SCLogDebug("flow %p direction %s frame %p starting at %" PRIu64 " len %" PRIi64
" (offset %" PRIu64 ")",
f, dir == 0 ? "toserver" : "toclient", frame_start, offset, len, stream_slice->offset);
#endif
BUG_ON(f->alparser == NULL);
FramesContainer *frames_container = AppLayerFramesSetupContainer(f);
if (frames_container == NULL)
return NULL;
TcpStream *stream;
TcpSession *ssn = f->protoctx;
Frames *frames;
if (dir == 0) {
frames = &frames_container->toserver;
stream = &ssn->client;
} else {
frames = &frames_container->toclient;
stream = &ssn->server;
}
int64_t abs_frame_offset = stream_slice->offset + (int64_t)ptr_offset;
int64_t rel_offset = abs_frame_offset - STREAM_BASE_OFFSET(stream);
Frame *r = FrameNew(frames, rel_offset, len);
if (r != NULL) {
r->type = frame_type;
}
return r;
}
static Frame *AppLayerFrameUdp(Flow *f, const StreamSlice *stream_slice,
const uint32_t frame_start_rel, const int64_t len, int dir, uint8_t frame_type)
{
BUG_ON(f->proto != IPPROTO_UDP);
FramesContainer *frames_container = AppLayerFramesSetupContainer(f);
if (frames_container == NULL)
return NULL;
Frames *frames;
if (dir == 0) {
frames = &frames_container->toserver;
} else {
frames = &frames_container->toclient;
}
Frame *r = FrameNew(frames, frame_start_rel, len);
if (r != NULL) {
r->type = frame_type;
}
return r;
}
/** \brief create new frame using a relative offset from the start of the stream slice
*/
Frame *AppLayerFrameNewByRelativeOffset(Flow *f, const StreamSlice *stream_slice,
const uint32_t frame_start_rel, const int64_t len, int dir, uint8_t frame_type)
{
/* workarounds for many (unit|fuzz)tests not handling TCP data properly */
#if defined(UNITTESTS) || defined(FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION)
if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
return NULL;
if (stream_slice->input == NULL)
return NULL;
#endif
BUG_ON(stream_slice->input == NULL);
BUG_ON(f->proto == IPPROTO_TCP && f->protoctx == NULL);
BUG_ON(f->alparser == NULL);
if (f->proto == IPPROTO_UDP) {
return AppLayerFrameUdp(f, stream_slice, frame_start_rel, len, dir, frame_type);
}
FramesContainer *frames_container = AppLayerFramesSetupContainer(f);
if (frames_container == NULL)
return NULL;
TcpStream *stream;
TcpSession *ssn = f->protoctx;
Frames *frames;
if (dir == 0) {
frames = &frames_container->toserver;
stream = &ssn->client;
} else {
frames = &frames_container->toclient;
stream = &ssn->server;
}
const uint64_t base = STREAM_BASE_OFFSET(stream);
#ifdef DEBUG
const uint64_t app = STREAM_APP_PROGRESS(stream);
const uint64_t app_offset = app - base;
const uint64_t slice_offset = stream_slice->offset - base;
SCLogDebug("app %" PRIu64 ", base %" PRIu64 ", slice %" PRIu64, app, base, slice_offset);
SCLogDebug("app_offset %" PRIu64 ", slice_offset %" PRIu64, app_offset, slice_offset);
#endif
const uint64_t frame_abs_offset = (uint64_t)frame_start_rel + stream_slice->offset;
const uint64_t frame_base_offset = frame_abs_offset - base;
SCLogDebug("frame_start_rel %u frame_abs_offset %" PRIu64 ", frame_base_offset %" PRIu64,
frame_start_rel, frame_abs_offset, frame_base_offset);
int64_t rel_offset = frame_base_offset;
#ifdef DEBUG
const char *type_name = AppLayerParserGetFrameNameById(f->proto, f->alproto, frame_type);
SCLogDebug("flow %p direction %s frame offset %u rel_offset %" PRIi64 " (abs %" PRIu64
") starting at %" PRIu64 " len %" PRIi64 " (offset %" PRIu64 ") type %u/%s",
f, dir == 0 ? "toserver" : "toclient", frame_start_rel, rel_offset, frame_abs_offset,
frame_abs_offset, len, stream_slice->offset, frame_type, type_name);
#endif
Frame *r = FrameNew(frames, rel_offset, len);
if (r != NULL) {
r->type = frame_type;
}
return r;
}
void AppLayerFrameDump(Flow *f)
{
if (f->proto == IPPROTO_TCP && f->protoctx && f->alparser) {
FramesContainer *frames_container = AppLayerFramesGetContainer(f);
if (frames_container != NULL) {
AppLayerFrameDumpForFrames("toserver::dump", &frames_container->toserver);
AppLayerFrameDumpForFrames("toclient::dump", &frames_container->toclient);
}
}
}
/** \brief create new frame using the absolute offset from the start of the stream
*/
Frame *AppLayerFrameNewByAbsoluteOffset(Flow *f, const StreamSlice *stream_slice,
const uint64_t frame_start, const int64_t len, int dir, uint8_t frame_type)
{
/* workarounds for many (unit|fuzz)tests not handling TCP data properly */
#if defined(UNITTESTS) || defined(FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION)
if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
return NULL;
if (stream_slice->input == NULL)
return NULL;
#endif
BUG_ON(stream_slice->input == NULL);
BUG_ON(f->proto == IPPROTO_TCP && f->protoctx == NULL);
BUG_ON(f->alparser == NULL);
BUG_ON(frame_start < stream_slice->offset);
BUG_ON(frame_start - stream_slice->offset >= (uint64_t)INT_MAX);
FramesContainer *frames_container = AppLayerFramesSetupContainer(f);
if (frames_container == NULL)
return NULL;
TcpSession *ssn = f->protoctx;
TcpStream *stream;
Frames *frames;
if (dir == 0) {
stream = &ssn->client;
frames = &frames_container->toserver;
} else {
stream = &ssn->server;
frames = &frames_container->toclient;
}
const uint64_t frame_start_rel = frame_start - STREAM_BASE_OFFSET(stream);
#ifdef DEBUG
SCLogDebug("flow %p direction %s frame offset %" PRIu64 " (abs %" PRIu64
") starting at %" PRIu64 " len %" PRIi64 " (offset %" PRIu64 ")",
f, dir == 0 ? "toserver" : "toclient", frame_start_rel, frame_start, frame_start, len,
stream_slice->offset);
#endif
Frame *r = FrameNew(frames, (uint32_t)frame_start_rel, len);
if (r != NULL) {
r->type = frame_type;
}
return r;
}
void AppLayerFrameAddEvent(Frame *r, uint8_t e)
{
if (r != NULL) {
if (r->event_cnt < 4) { // TODO
r->events[r->event_cnt++] = e;
}
FrameDebug("add_event", NULL, r);
}
}
void AppLayerFrameAddEventById(Flow *f, const int dir, const FrameId id, uint8_t e)
{
Frame *frame = AppLayerFrameGetById(f, dir, id);
AppLayerFrameAddEvent(frame, e);
}
FrameId AppLayerFrameGetId(Frame *r)
{
if (r != NULL) {
return r->id;
} else {
return -1;
}
}
void AppLayerFrameSetLength(Frame *frame, int64_t len)
{
if (frame != NULL) {
frame->len = len;
FrameDebug("set_length", NULL, frame);
}
}
void AppLayerFrameSetLengthById(Flow *f, const int dir, const FrameId id, int64_t len)
{
Frame *frame = AppLayerFrameGetById(f, dir, id);
AppLayerFrameSetLength(frame, len);
}
void AppLayerFrameSetTxId(Frame *r, uint64_t tx_id)
{
if (r != NULL) {
r->flags |= FRAME_FLAG_TX_ID_SET;
r->tx_id = tx_id;
FrameDebug("set_txid", NULL, r);
}
}
void AppLayerFrameSetTxIdById(Flow *f, const int dir, const FrameId id, uint64_t tx_id)
{
Frame *frame = AppLayerFrameGetById(f, dir, id);
AppLayerFrameSetTxId(frame, tx_id);
}
Frame *AppLayerFrameGetById(Flow *f, const int dir, const FrameId frame_id)
{
FramesContainer *frames_container = AppLayerFramesGetContainer(f);
if (frames_container == NULL)
return NULL;
Frames *frames;
if (dir == 0) {
frames = &frames_container->toserver;
} else {
frames = &frames_container->toclient;
}
return FrameGetById(frames, frame_id);
}
static inline bool FrameIsDone(
const Frame *frame, const uint64_t abs_offset, const uint64_t abs_right_edge)
{
/* frame with negative length means we don't know the size yet. */
if (frame->len < 0)
return false;
const int64_t frame_abs_offset = (int64_t)abs_offset + frame->rel_offset;
const int64_t frame_right_edge = frame_abs_offset + frame->len;
if ((uint64_t)frame_right_edge <= abs_right_edge) {
SCLogDebug("frame %p id %" PRIi64 " is done", frame, frame->id);
return true;
}
return false;
}
static void FramePrune(Frames *frames, const TcpStream *stream, const bool eof)
{
const uint64_t frames_le_start = (uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream);
SCLogDebug("start: left edge %" PRIu64 ", left_edge_rel %u, stream base %" PRIu64,
(uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream), frames->left_edge_rel,
STREAM_BASE_OFFSET(stream));
const uint64_t abs_offset = STREAM_BASE_OFFSET(stream);
const uint64_t acked = StreamTcpGetUsable(stream, eof);
uint64_t le = STREAM_APP_PROGRESS(stream);
const uint16_t start = frames->cnt;
uint16_t removed = 0;
uint16_t x = 0;
for (uint16_t i = 0; i < frames->cnt; i++) {
if (i < FRAMES_STATIC_CNT) {
Frame *frame = &frames->sframes[i];
FrameDebug("prune(s)", frames, frame);
if (eof || FrameIsDone(frame, abs_offset, acked)) {
// remove by not incrementing 'x'
SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
FrameDebug("remove(s)", frames, frame);
FrameClean(frame);
removed++;
} else {
const uint64_t fle = FrameLeftEdge(stream, frame, STREAM_BASE_OFFSET(stream));
le = MIN(le, fle);
SCLogDebug("le %" PRIu64 ", frame fle %" PRIu64, le, fle);
Frame *nframe = &frames->sframes[x];
FrameCopy(nframe, frame);
if (frame != nframe) {
FrameClean(frame);
}
x++;
}
} else {
const uint16_t o = i - FRAMES_STATIC_CNT;
Frame *frame = &frames->dframes[o];
FrameDebug("prune(d)", frames, frame);
if (eof || FrameIsDone(frame, abs_offset, acked)) {
// remove by not incrementing 'x'
SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
FrameDebug("remove(d)", frames, frame);
FrameClean(frame);
removed++;
} else {
const uint64_t fle = FrameLeftEdge(stream, frame, STREAM_BASE_OFFSET(stream));
le = MIN(le, fle);
SCLogDebug("le %" PRIu64 ", frame fle %" PRIu64, le, fle);
Frame *nframe;
if (x >= FRAMES_STATIC_CNT) {
nframe = &frames->dframes[x - FRAMES_STATIC_CNT];
} else {
nframe = &frames->sframes[x];
}
FrameCopy(nframe, frame);
if (frame != nframe) {
FrameClean(frame);
}
x++;
}
}
}
frames->cnt = x;
frames->left_edge_rel = le - STREAM_BASE_OFFSET(stream);
#ifdef DEBUG
SCLogDebug("end: left edge %" PRIu64 ", left_edge_rel %u, stream base %" PRIu64
", cnt %u, removed %u, start %u",
(uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream), frames->left_edge_rel,
STREAM_BASE_OFFSET(stream), frames->cnt, removed, start);
AppLayerFrameDumpForFrames("post_slide", frames);
#endif
BUG_ON(le < STREAM_BASE_OFFSET(stream));
if (frames->cnt > 0) { // if we removed all this can fail
BUG_ON(frames_le_start > le);
}
BUG_ON(x != start - removed);
}
void FramesPrune(Flow *f, Packet *p)
{
if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
return;
FramesContainer *frames_container = AppLayerFramesGetContainer(f);
if (frames_container == NULL)
return;
Frames *frames;
if (p->proto == IPPROTO_UDP) {
SCLogDebug("clearing all UDP frames");
if (PKT_IS_TOSERVER(p)) {
frames = &frames_container->toserver;
} else {
frames = &frames_container->toclient;
}
FramesClear(frames);
return;
}
TcpSession *ssn = f->protoctx;
if (ssn->flags & STREAMTCP_FLAG_APP_LAYER_DISABLED) {
AppLayerFramesFreeContainer(f);
return;
}
TcpStream *stream;
if (PKT_IS_TOSERVER(p)) {
stream = &ssn->client;
frames = &frames_container->toserver;
} else {
stream = &ssn->server;
frames = &frames_container->toclient;
}
const bool eof = ssn->state == TCP_CLOSED || PKT_IS_PSEUDOPKT(p);
SCLogDebug("eof %s", eof ? "TRUE" : "false");
FramePrune(frames, stream, eof);
}