• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

OISF / suricata / 23374838686

21 Mar 2026 07:29AM UTC coverage: 59.341% (-20.0%) from 79.315%
23374838686

Pull #15075

github

web-flow
Merge 90b4e834f into 6587e363a
Pull Request #15075: Stack 8001 v16.4

38 of 70 new or added lines in 10 files covered. (54.29%)

34165 existing lines in 563 files now uncovered.

119621 of 201584 relevant lines covered (59.34%)

650666.92 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

94.82
/src/app-layer-frames.c
1
/* Copyright (C) 2007-2024 Open Information Security Foundation
2
 *
3
 * You can copy, redistribute or modify this Program under the terms of
4
 * the GNU General Public License version 2 as published by the Free
5
 * Software Foundation.
6
 *
7
 * This program is distributed in the hope that it will be useful,
8
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10
 * GNU General Public License for more details.
11
 *
12
 * You should have received a copy of the GNU General Public License
13
 * version 2 along with this program; if not, write to the Free Software
14
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
15
 * 02110-1301, USA.
16
 */
17

18
/**
19
 * \file
20
 *
21
 * \author Victor Julien <victor@inliniac.net>
22
 *
23
 */
24

25
#include "suricata-common.h"
26
#include "util-print.h"
27

28
#include "flow.h"
29
#include "stream-tcp.h"
30
#include "rust.h"
31
#include "app-layer-frames.h"
32
#include "app-layer-parser.h"
33

34
struct FrameConfig {
35
    SC_ATOMIC_DECLARE(uint64_t, types);
36
};
37
/* This array should be allocated to contain g_alproto_max protocols. */
38
static struct FrameConfig *frame_config;
39

40
void FrameConfigInit(void)
41
{
4✔
42
    frame_config = SCCalloc(g_alproto_max, sizeof(struct FrameConfig));
4✔
43
    if (unlikely(frame_config == NULL)) {
4✔
44
        FatalError("Unable to alloc frame_config.");
×
45
    }
×
46
    for (AppProto p = 0; p < g_alproto_max; p++) {
164✔
47
        SC_ATOMIC_INIT(frame_config[p].types);
160✔
48
    }
160✔
49
}
4✔
50

51
void FrameConfigDeInit(void)
UNCOV
52
{
×
UNCOV
53
    SCFree(frame_config);
×
UNCOV
54
}
×
55

56
void FrameConfigEnableAll(void)
UNCOV
57
{
×
UNCOV
58
    const uint64_t bits = UINT64_MAX;
×
UNCOV
59
    for (AppProto p = 0; p < g_alproto_max; p++) {
×
UNCOV
60
        struct FrameConfig *fc = &frame_config[p];
×
UNCOV
61
        SC_ATOMIC_OR(fc->types, bits);
×
UNCOV
62
    }
×
UNCOV
63
}
×
64

65
void FrameConfigEnable(const AppProto p, const uint8_t type)
66
{
36,459✔
67
    const uint64_t bits = BIT_U64(type);
36,459✔
68
    struct FrameConfig *fc = &frame_config[p];
36,459✔
69
    SC_ATOMIC_OR(fc->types, bits);
36,459✔
70
}
36,459✔
71

72
static inline bool FrameConfigTypeIsEnabled(const AppProto p, const uint8_t type)
73
{
2,787,357✔
74
    struct FrameConfig *fc = &frame_config[p];
2,787,357✔
75
    const uint64_t bits = BIT_U64(type);
2,787,357✔
76
    const bool enabled = (SC_ATOMIC_GET(fc->types) & bits) != 0;
2,787,357✔
77
    return enabled;
2,787,357✔
78
}
2,787,357✔
79

80
#ifdef DEBUG
81
static void FrameDebug(const char *prefix, const Frames *frames, const Frame *frame)
82
{
83
    const char *type_name = "unknown";
84
    if (frame->type == FRAME_STREAM_TYPE) {
85
        type_name = "stream";
86
    } else if (frames != NULL) {
87
        type_name = AppLayerParserGetFrameNameById(frames->ipproto, frames->alproto, frame->type);
88
    }
89
    SCLogDebug("[%s] %p: frame:%p type:%u/%s id:%" PRIi64 " flags:%02x offset:%" PRIu64
90
               ", len:%" PRIi64 ", inspect_progress:%" PRIu64 ", events:%u %u/%u/%u/%u",
91
            prefix, frames, frame, frame->type, type_name, frame->id, frame->flags, frame->offset,
92
            frame->len, frame->inspect_progress, frame->event_cnt, frame->events[0],
93
            frame->events[1], frame->events[2], frame->events[3]);
94
}
95
#else
96
#define FrameDebug(prefix, frames, frame)
97
#endif
98

99
/**
100
 * \note "open" means a frame that has no length set (len == -1)
101
 * \todo perhaps we can search backwards */
102
Frame *FrameGetLastOpenByType(Frames *frames, const uint8_t frame_type)
103
{
311,139✔
104
    Frame *candidate = NULL;
311,139✔
105

106
    SCLogDebug(
311,139✔
107
            "frames %p cnt %u, looking for last of type %" PRIu8, frames, frames->cnt, frame_type);
311,139✔
108
    for (uint16_t i = 0; i < frames->cnt; i++) {
1,551,611✔
109
        if (i < FRAMES_STATIC_CNT) {
1,240,472✔
110
            Frame *frame = &frames->sframes[i];
605,864✔
111
            FrameDebug("get_by_id(static)", frames, frame);
605,864✔
112
            if (frame->type == frame_type && frame->len == -1)
605,864✔
113
                candidate = frame;
8,095✔
114
        } else {
634,608✔
115
            const uint16_t o = i - FRAMES_STATIC_CNT;
634,608✔
116
            Frame *frame = &frames->dframes[o];
634,608✔
117
            FrameDebug("get_by_id(dynamic)", frames, frame);
634,608✔
118
            if (frame->type == frame_type && frame->len == -1)
634,608✔
119
                candidate = frame;
15✔
120
        }
634,608✔
121
    }
1,240,472✔
122
    return candidate;
311,139✔
123
}
311,139✔
124

125
Frame *FrameGetById(Frames *frames, const int64_t id)
126
{
146,899✔
127
    SCLogDebug("frames %p cnt %u, looking for %" PRIi64, frames, frames->cnt, id);
146,899✔
128
    for (uint16_t i = 0; i < frames->cnt; i++) {
1,956,873✔
129
        if (i < FRAMES_STATIC_CNT) {
1,956,763✔
130
            Frame *frame = &frames->sframes[i];
361,616✔
131
            FrameDebug("get_by_id(static)", frames, frame);
361,616✔
132
            if (frame->id == id)
361,616✔
133
                return frame;
67,481✔
134
        } else {
1,595,147✔
135
            const uint16_t o = i - FRAMES_STATIC_CNT;
1,595,147✔
136
            Frame *frame = &frames->dframes[o];
1,595,147✔
137
            FrameDebug("get_by_id(dynamic)", frames, frame);
1,595,147✔
138
            if (frame->id == id)
1,595,147✔
139
                return frame;
79,308✔
140
        }
1,595,147✔
141
    }
1,956,763✔
142
    return NULL;
110✔
143
}
146,899✔
144

145
Frame *FrameGetByIndex(Frames *frames, const uint32_t idx)
146
{
11,163,163✔
147
    if (idx >= frames->cnt)
11,163,163✔
148
        return NULL;
×
149

150
    if (idx < FRAMES_STATIC_CNT) {
11,163,163✔
151
        Frame *frame = &frames->sframes[idx];
795,865✔
152
        FrameDebug("get_by_idx(s)", frames, frame);
795,865✔
153
        return frame;
795,865✔
154
    } else {
10,367,298✔
155
        const uint32_t o = idx - FRAMES_STATIC_CNT;
10,367,298✔
156
        Frame *frame = &frames->dframes[o];
10,367,298✔
157
        FrameDebug("get_by_idx(d)", frames, frame);
10,367,298✔
158
        return frame;
10,367,298✔
159
    }
10,367,298✔
160
}
11,163,163✔
161

162
static Frame *FrameNew(Frames *frames, uint64_t offset, int64_t len)
163
{
1,242,023✔
164
    DEBUG_VALIDATE_BUG_ON(frames == NULL);
1,242,023✔
165

166
    if (frames->cnt < FRAMES_STATIC_CNT) {
1,242,023✔
167
        Frame *frame = &frames->sframes[frames->cnt];
721,924✔
168
        frames->sframes[frames->cnt].offset = offset;
721,924✔
169
        frames->sframes[frames->cnt].len = len;
721,924✔
170
        frames->sframes[frames->cnt].id = ++frames->base_id;
721,924✔
171
        frames->cnt++;
721,924✔
172
        return frame;
721,924✔
173
    } else if (frames->dframes == NULL) {
721,924✔
174
        DEBUG_VALIDATE_BUG_ON(frames->dyn_size != 0);
8,820✔
175
        DEBUG_VALIDATE_BUG_ON(frames->cnt != FRAMES_STATIC_CNT);
8,820✔
176

177
        frames->dframes = SCCalloc(8, sizeof(Frame));
8,820✔
178
        if (frames->dframes == NULL) {
8,820✔
179
            return NULL;
×
180
        }
×
181
        frames->cnt++;
8,820✔
182
        DEBUG_VALIDATE_BUG_ON(frames->cnt != FRAMES_STATIC_CNT + 1);
8,820✔
183

184
        frames->dyn_size = 8;
8,820✔
185
        frames->dframes[0].offset = offset;
8,820✔
186
        frames->dframes[0].len = len;
8,820✔
187
        frames->dframes[0].id = ++frames->base_id;
8,820✔
188
        return &frames->dframes[0];
8,820✔
189
    } else {
511,279✔
190
        DEBUG_VALIDATE_BUG_ON(frames->cnt < FRAMES_STATIC_CNT);
511,279✔
191

192
        /* need to handle dynamic storage of frames now */
193
        const uint16_t dyn_cnt = frames->cnt - FRAMES_STATIC_CNT;
511,279✔
194
        if (dyn_cnt < frames->dyn_size) {
511,279✔
195
            DEBUG_VALIDATE_BUG_ON(frames->dframes == NULL);
401,719✔
196

197
            // fall through
198
        } else {
401,719✔
199
            if (frames->dyn_size == 256) {
109,560✔
200
                SCLogDebug("limit reached! 256 dynamic frames already");
102,097✔
201
                // limit reached
202
                // TODO figure out if this should lead to an event of sorts
203
                return NULL;
102,097✔
204
            }
102,097✔
205

206
            /* realloc time */
207
            uint16_t new_dyn_size = frames->dyn_size * 2;
7,463✔
208
            uint32_t new_alloc_size = new_dyn_size * sizeof(Frame);
7,463✔
209

210
            void *ptr = SCRealloc(frames->dframes, new_alloc_size);
7,463✔
211
            if (ptr == NULL) {
7,463✔
212
                return NULL;
×
213
            }
×
214

215
            memset((uint8_t *)ptr + (frames->dyn_size * sizeof(Frame)), 0x00,
7,463✔
216
                    (frames->dyn_size * sizeof(Frame)));
7,463✔
217
            frames->dframes = ptr;
7,463✔
218
            frames->dyn_size = new_dyn_size;
7,463✔
219
        }
7,463✔
220

221
        frames->cnt++;
409,182✔
222
        frames->dframes[dyn_cnt].offset = offset;
409,182✔
223
        frames->dframes[dyn_cnt].len = len;
409,182✔
224
        frames->dframes[dyn_cnt].id = ++frames->base_id;
409,182✔
225
        return &frames->dframes[dyn_cnt];
409,182✔
226
    }
511,279✔
227
}
1,242,023✔
228

229
static void FrameClean(Frame *frame)
230
{
1,157,943✔
231
    memset(frame, 0, sizeof(*frame));
1,157,943✔
232
}
1,157,943✔
233

234
static void FrameCopy(Frame *dst, Frame *src)
235
{
13,985,963✔
236
    memcpy(dst, src, sizeof(*dst));
13,985,963✔
237
}
13,985,963✔
238

239
#ifdef DEBUG
240
static void AppLayerFrameDumpForFrames(const char *prefix, const Frames *frames)
241
{
242
    SCLogDebug("prefix: %s", prefix);
243
    for (uint16_t i = 0; i < frames->cnt; i++) {
244
        if (i < FRAMES_STATIC_CNT) {
245
            const Frame *frame = &frames->sframes[i];
246
            FrameDebug(prefix, frames, frame);
247
        } else {
248
            const uint16_t o = i - FRAMES_STATIC_CNT;
249
            const Frame *frame = &frames->dframes[o];
250
            FrameDebug(prefix, frames, frame);
251
        }
252
    }
253
    SCLogDebug("prefix: %s", prefix);
254
}
255
#endif
256

257
static inline uint64_t FrameLeftEdge(const TcpStream *stream, const Frame *frame)
258
{
14,421,534✔
259
    const int64_t app_progress = STREAM_APP_PROGRESS(stream);
14,421,534✔
260

261
    const int64_t frame_offset = frame->offset;
14,421,534✔
262
    const int64_t frame_data = app_progress - frame_offset;
14,421,534✔
263

264
    SCLogDebug("frame_offset %" PRIi64 ", frame_data %" PRIi64 ", frame->len %" PRIi64,
14,421,534✔
265
            frame_offset, frame_data, frame->len);
14,421,534✔
266
    DEBUG_VALIDATE_BUG_ON(frame_offset > app_progress);
14,421,534✔
267

268
    /* length unknown, make sure to have at least 2500 */
269
    if (frame->len < 0) {
14,421,534✔
270
        if (frame_data <= 2500) {
203,273✔
271
            SCLogDebug("got <= 2500 bytes (%" PRIu64 "), returning offset %" PRIu64, frame_data,
125,799✔
272
                    frame_offset);
125,799✔
273
            return frame_offset;
125,799✔
274
        } else {
125,799✔
275
            SCLogDebug("got > 2500 bytes (%" PRIu64 "), returning offset %" PRIu64, frame_data,
77,474✔
276
                    (frame_offset + (frame_data - 2500)));
77,474✔
277
            return frame_offset + (frame_data - 2500);
77,474✔
278
        }
77,474✔
279

280
        /* length specified */
281
    } else {
14,218,261✔
282
        /* have all data for the frame, we can skip it */
283
        if (frame->len <= frame_data) {
14,218,261✔
284
            uint64_t x = frame_offset + frame_data;
14,193,405✔
285
            SCLogDebug("x %" PRIu64, x);
14,193,405✔
286
            return x;
14,193,405✔
287
            /*
288

289
                [ stream      <frame_data> ]
290
                             [ frame        .......]
291

292
             */
293
        } else if (frame_data < 2500) {
14,193,405✔
294
            uint64_t x = frame_offset;
13,660✔
295
            SCLogDebug("x %" PRIu64, x);
13,660✔
296
            return x;
13,660✔
297
        } else {
13,660✔
298
            uint64_t x = frame_offset + (frame_data - 2500);
11,196✔
299
            SCLogDebug("x %" PRIu64, x);
11,196✔
300
            return x;
11,196✔
301
        }
11,196✔
302
    }
14,218,261✔
303
}
14,421,534✔
304

305
/** Stream buffer slides forward, we need to update and age out
306
 *  frame offsets/frames. Aging out means we move existing frames
307
 *  into the slots we'd free up.
308
 *
309
 *  Start:
310
 *
311
 *  [ stream ]
312
 *    [ frame   ...........]
313
 *      offset: 2
314
 *      len: 19
315
 *
316
 *  Slide:
317
 *         [ stream ]
318
 *    [ frame ....          .]
319
 *      offset: 2
320
 *       len: 19
321
 *
322
 *  Slide:
323
 *                [ stream ]
324
 *    [ frame ...........    ]
325
 *      offset: 2
326
 *      len: 19
327
 */
328
static int FrameSlide(const char *ds, Frames *frames, const TcpStream *stream, const uint32_t slide)
329
{
15,710✔
330
    SCLogDebug("start: left edge %" PRIu64 ", left_edge_rel %u, stream base %" PRIu64
15,710✔
331
               ", next %" PRIu64,
15,710✔
332
            (uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream), frames->left_edge_rel,
15,710✔
333
            STREAM_BASE_OFFSET(stream), STREAM_BASE_OFFSET(stream) + slide);
15,710✔
334
    DEBUG_VALIDATE_BUG_ON(frames == NULL);
15,710✔
335
    SCLogDebug("%s frames %p: sliding %u bytes", ds, frames, slide);
15,710✔
336
    uint64_t le = STREAM_APP_PROGRESS(stream);
15,710✔
337
    const uint64_t next_base = STREAM_BASE_OFFSET(stream) + slide;
15,710✔
338
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
339
    const uint16_t start = frames->cnt;
340
    uint16_t removed = 0;
341
#endif
342
    uint16_t x = 0;
15,710✔
343
    for (uint16_t i = 0; i < frames->cnt; i++) {
457,497✔
344
        if (i < FRAMES_STATIC_CNT) {
441,787✔
345
            Frame *frame = &frames->sframes[i];
11,311✔
346
            FrameDebug("slide(s)", frames, frame);
11,311✔
347
            if (frame->len >= 0 && frame->offset + frame->len <= next_base) {
11,311✔
348
                // remove by not incrementing 'x'
349
                SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
189✔
350
                FrameClean(frame);
189✔
351
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
352
                removed++;
353
#endif
354
            } else {
11,122✔
355
                Frame *nframe = &frames->sframes[x];
11,122✔
356
                FrameCopy(nframe, frame);
11,122✔
357
                if (frame != nframe) {
11,122✔
358
                    FrameClean(frame);
10✔
359
                }
10✔
360
                le = MIN(le, FrameLeftEdge(stream, nframe));
11,122✔
361
                x++;
11,122✔
362
            }
11,122✔
363
        } else {
430,476✔
364
            const uint16_t o = i - FRAMES_STATIC_CNT;
430,476✔
365
            Frame *frame = &frames->dframes[o];
430,476✔
366
            FrameDebug("slide(d)", frames, frame);
430,476✔
367
            if (frame->len >= 0 && frame->offset + frame->len <= next_base) {
430,476✔
368
                // remove by not incrementing 'x'
369
                SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
5,837✔
370
                FrameClean(frame);
5,837✔
371
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
372
                removed++;
373
#endif
374
            } else {
424,639✔
375
                Frame *nframe;
424,639✔
376
                if (x >= FRAMES_STATIC_CNT) {
424,639✔
377
                    nframe = &frames->dframes[x - FRAMES_STATIC_CNT];
424,478✔
378
                } else {
424,478✔
379
                    nframe = &frames->sframes[x];
161✔
380
                }
161✔
381
                FrameCopy(nframe, frame);
424,639✔
382
                if (frame != nframe) {
424,639✔
383
                    FrameClean(frame);
6,323✔
384
                }
6,323✔
385
                le = MIN(le, FrameLeftEdge(stream, nframe));
424,639✔
386
                x++;
424,639✔
387
            }
424,639✔
388
        }
430,476✔
389
    }
441,787✔
390
    frames->cnt = x;
15,710✔
391
    uint64_t o = STREAM_BASE_OFFSET(stream) + slide;
15,710✔
392
    DEBUG_VALIDATE_BUG_ON(o > le);
15,710✔
393
    DEBUG_VALIDATE_BUG_ON(le - o > UINT32_MAX);
15,710✔
394
    frames->left_edge_rel = (uint32_t)(le - o);
15,710✔
395

396
#ifdef DEBUG
397
    SCLogDebug("end: left edge %" PRIu64 ", left_edge_rel %u, stream base %" PRIu64
398
               " (+slide), cnt %u, removed %u, start %u",
399
            (uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream) + slide,
400
            frames->left_edge_rel, STREAM_BASE_OFFSET(stream) + slide, frames->cnt, removed, start);
401
    char pf[32] = "";
402
    snprintf(pf, sizeof(pf), "%s:post_slide", ds);
403
    AppLayerFrameDumpForFrames(pf, frames);
404
#endif
405
    DEBUG_VALIDATE_BUG_ON(x != start - removed);
15,710✔
406
    return 0;
15,710✔
407
}
15,710✔
408

409
void AppLayerFramesSlide(Flow *f, const uint32_t slide, const uint8_t direction)
410
{
16,296✔
411
    FramesContainer *frames_container = AppLayerFramesGetContainer(f);
16,296✔
412
    if (frames_container == NULL)
16,296✔
413
        return;
586✔
414
    Frames *frames;
15,710✔
415
    TcpSession *ssn = f->protoctx;
15,710✔
416
    TcpStream *stream;
15,710✔
417
    if (direction == STREAM_TOSERVER) {
15,710✔
418
        stream = &ssn->client;
6,997✔
419
        frames = &frames_container->toserver;
6,997✔
420
        FrameSlide("toserver", frames, stream, slide);
6,997✔
421
    } else {
8,713✔
422
        stream = &ssn->server;
8,713✔
423
        frames = &frames_container->toclient;
8,713✔
424
        FrameSlide("toclient", frames, stream, slide);
8,713✔
425
    }
8,713✔
426
}
15,710✔
427

428
static void FrameFreeSingleFrame(Frames *frames, Frame *r)
429
{
180,437✔
430
    FrameDebug("free", frames, r);
180,437✔
431
    FrameClean(r);
180,437✔
432
}
180,437✔
433

434
static void FramesClear(Frames *frames)
435
{
81,539✔
436
    DEBUG_VALIDATE_BUG_ON(frames == NULL);
81,539✔
437

438
    SCLogDebug("frames %u", frames->cnt);
81,539✔
439
    for (uint16_t i = 0; i < frames->cnt; i++) {
261,976✔
440
        if (i < FRAMES_STATIC_CNT) {
180,437✔
441
            Frame *r = &frames->sframes[i];
65,867✔
442
            SCLogDebug("removing frame %p", r);
65,867✔
443
            FrameFreeSingleFrame(frames, r);
65,867✔
444
        } else {
114,570✔
445
            const uint16_t o = i - FRAMES_STATIC_CNT;
114,570✔
446
            Frame *r = &frames->dframes[o];
114,570✔
447
            SCLogDebug("removing frame %p", r);
114,570✔
448
            FrameFreeSingleFrame(frames, r);
114,570✔
449
        }
114,570✔
450
    }
180,437✔
451
    frames->cnt = 0;
81,539✔
452
}
81,539✔
453

454
void FramesFree(Frames *frames)
455
{
55,228✔
456
    DEBUG_VALIDATE_BUG_ON(frames == NULL);
55,228✔
457
    FramesClear(frames);
55,228✔
458
    SCFree(frames->dframes);
55,228✔
459
    frames->dframes = NULL;
55,228✔
460
}
55,228✔
461

462
/** \brief create new frame using a pointer to start of the frame
463
 */
464
Frame *AppLayerFrameNewByPointer(Flow *f, const StreamSlice *stream_slice,
465
        const uint8_t *frame_start, const int64_t len, int dir, uint8_t frame_type)
466
{
126,874✔
467
    SCLogDebug("frame_start:%p stream_slice->input:%p stream_slice->offset:%" PRIu64, frame_start,
126,874✔
468
            stream_slice->input, stream_slice->offset);
126,874✔
469

470
    if (!(FrameConfigTypeIsEnabled(f->alproto, frame_type)))
126,874✔
471
        return NULL;
73,813✔
472

473
        /* workarounds for many (unit|fuzz)tests not handling TCP data properly */
474
#if defined(UNITTESTS) || defined(FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION)
53,061✔
475
    if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
53,061✔
UNCOV
476
        return NULL;
×
477
    if (frame_start < stream_slice->input ||
53,061✔
478
            frame_start > stream_slice->input + stream_slice->input_len)
53,061✔
UNCOV
479
        return NULL;
×
480
#endif
53,061✔
481
    DEBUG_VALIDATE_BUG_ON(frame_start < stream_slice->input);
53,061✔
482
    DEBUG_VALIDATE_BUG_ON(stream_slice->input == NULL);
53,061✔
483
    DEBUG_VALIDATE_BUG_ON(f->proto == IPPROTO_TCP && f->protoctx == NULL);
53,061✔
484

485
    ptrdiff_t ptr_offset = frame_start - stream_slice->input;
53,061✔
486
#ifdef DEBUG
487
    uint64_t offset = ptr_offset + stream_slice->offset;
488
    SCLogDebug("flow %p direction %s frame %p starting at %" PRIu64 " len %" PRIi64
489
               " (offset %" PRIu64 ")",
490
            f, dir == 0 ? "toserver" : "toclient", frame_start, offset, len, stream_slice->offset);
491
#endif
492
    DEBUG_VALIDATE_BUG_ON(f->alparser == NULL);
53,061✔
493

494
    FramesContainer *frames_container = AppLayerFramesSetupContainer(f);
53,061✔
495
    if (frames_container == NULL)
53,061✔
496
        return NULL;
×
497

498
    Frames *frames;
53,061✔
499
    if (dir == 0) {
53,061✔
500
        frames = &frames_container->toserver;
24,696✔
501
    } else {
28,365✔
502
        frames = &frames_container->toclient;
28,365✔
503
    }
28,365✔
504

505
    uint64_t abs_frame_offset = stream_slice->offset + ptr_offset;
53,061✔
506

507
    Frame *r = FrameNew(frames, abs_frame_offset, len);
53,061✔
508
    if (r != NULL) {
53,061✔
509
        r->type = frame_type;
52,055✔
510
        FrameDebug("new_by_ptr", frames, r);
52,055✔
511
    }
52,055✔
512
    return r;
53,061✔
513
}
53,061✔
514

515
static Frame *AppLayerFrameUdp(
516
        Flow *f, const uint32_t frame_start_rel, const int64_t len, int dir, uint8_t frame_type)
517
{
31,780✔
518
    DEBUG_VALIDATE_BUG_ON(f->proto != IPPROTO_UDP);
31,780✔
519

520
    if (!(FrameConfigTypeIsEnabled(f->alproto, frame_type)))
31,780✔
521
        return NULL;
×
522

523
    FramesContainer *frames_container = AppLayerFramesSetupContainer(f);
31,780✔
524
    if (frames_container == NULL)
31,780✔
525
        return NULL;
×
526

527
    Frames *frames;
31,780✔
528
    if (dir == 0) {
31,780✔
529
        frames = &frames_container->toserver;
25,304✔
530
    } else {
25,304✔
531
        frames = &frames_container->toclient;
6,476✔
532
    }
6,476✔
533

534
    Frame *r = FrameNew(frames, frame_start_rel, len);
31,780✔
535
    if (r != NULL) {
31,780✔
536
        r->type = frame_type;
31,780✔
537
    }
31,780✔
538
    return r;
31,780✔
539
}
31,780✔
540

541
/** \brief create new frame using a relative offset from the start of the stream slice
542
 */
543
Frame *SCAppLayerFrameNewByRelativeOffset(Flow *f, const void *ss, const uint32_t frame_start_rel,
544
        const int64_t len, int dir, uint8_t frame_type)
545
{
2,210,483✔
546
    // need to hide StreamSlice argument
547
    // as we cannot bindgen a C function with an argument whose type
548
    // is defined in rust (at least before a suricata_core crate)
549
    const StreamSlice *stream_slice = (const StreamSlice *)ss;
2,210,483✔
550
    if (!(FrameConfigTypeIsEnabled(f->alproto, frame_type)))
2,210,483✔
551
        return NULL;
1,064,593✔
552

553
        /* workarounds for many (unit|fuzz)tests not handling TCP data properly */
554
#if defined(UNITTESTS) || defined(FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION)
1,145,890✔
555
    if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
1,145,890✔
UNCOV
556
        return NULL;
×
557
    if (stream_slice->input == NULL)
1,145,890✔
UNCOV
558
        return NULL;
×
559
#else
560
    DEBUG_VALIDATE_BUG_ON(stream_slice->input == NULL);
561
#endif
562
    DEBUG_VALIDATE_BUG_ON(f->proto == IPPROTO_TCP && f->protoctx == NULL);
1,145,890✔
563
    DEBUG_VALIDATE_BUG_ON(f->alparser == NULL);
1,145,890✔
564

565
    if (f->proto == IPPROTO_UDP) {
1,145,890✔
566
        return AppLayerFrameUdp(f, frame_start_rel, len, dir, frame_type);
31,780✔
567
    }
31,780✔
568

569
    FramesContainer *frames_container = AppLayerFramesSetupContainer(f);
1,114,110✔
570
    if (frames_container == NULL)
1,114,110✔
571
        return NULL;
×
572

573
    Frames *frames;
1,114,110✔
574
    if (dir == 0) {
1,114,110✔
575
        frames = &frames_container->toserver;
554,587✔
576
    } else {
559,523✔
577
        frames = &frames_container->toclient;
559,523✔
578
    }
559,523✔
579

580
    const uint64_t frame_abs_offset = (uint64_t)frame_start_rel + stream_slice->offset;
1,114,110✔
581
#ifdef DEBUG_VALIDATION
582
    const TcpSession *ssn = f->protoctx;
583
    const TcpStream *stream = dir == 0 ? &ssn->client : &ssn->server;
584
    BUG_ON(stream_slice->offset != STREAM_APP_PROGRESS(stream));
585
    BUG_ON(frame_abs_offset > STREAM_APP_PROGRESS(stream) + stream_slice->input_len);
586
#endif
587
    Frame *r = FrameNew(frames, frame_abs_offset, len);
1,114,110✔
588
    if (r != NULL) {
1,114,110✔
589
        r->type = frame_type;
1,013,019✔
590
    }
1,013,019✔
591
    return r;
1,114,110✔
592
}
1,114,110✔
593

594
void AppLayerFrameDump(Flow *f)
595
{
494,811✔
596
#ifdef DEBUG
597
    if (f->proto == IPPROTO_TCP && f->protoctx && f->alparser) {
598
        FramesContainer *frames_container = AppLayerFramesGetContainer(f);
599
        if (frames_container != NULL) {
600
            AppLayerFrameDumpForFrames("toserver::dump", &frames_container->toserver);
601
            AppLayerFrameDumpForFrames("toclient::dump", &frames_container->toclient);
602
        }
603
    }
604
#endif
605
}
494,811✔
606

607
/** \brief create new frame using the absolute offset from the start of the stream
608
 */
609
Frame *AppLayerFrameNewByAbsoluteOffset(Flow *f, const StreamSlice *stream_slice,
610
        const uint64_t frame_start, const int64_t len, int dir, uint8_t frame_type)
611
{
65,769✔
612
    if (!(FrameConfigTypeIsEnabled(f->alproto, frame_type)))
65,769✔
613
        return NULL;
22,696✔
614

615
        /* workarounds for many (unit|fuzz)tests not handling TCP data properly */
616
#if defined(UNITTESTS) || defined(FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION)
43,073✔
617
    if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
43,073✔
UNCOV
618
        return NULL;
×
619
    if (stream_slice->input == NULL)
43,073✔
620
        return NULL;
1✔
621
#else
622
    DEBUG_VALIDATE_BUG_ON(stream_slice->input == NULL);
623
#endif
624
    DEBUG_VALIDATE_BUG_ON(f->proto == IPPROTO_TCP && f->protoctx == NULL);
43,072✔
625
    DEBUG_VALIDATE_BUG_ON(f->alparser == NULL);
43,072✔
626
    DEBUG_VALIDATE_BUG_ON(frame_start < stream_slice->offset);
43,072✔
627
    DEBUG_VALIDATE_BUG_ON(frame_start - stream_slice->offset >= (uint64_t)INT_MAX);
43,072✔
628

629
    FramesContainer *frames_container = AppLayerFramesSetupContainer(f);
43,072✔
630
    if (frames_container == NULL)
43,072✔
631
        return NULL;
×
632

633
    Frames *frames;
43,072✔
634
    if (dir == 0) {
43,072✔
635
        frames = &frames_container->toserver;
22,861✔
636
    } else {
22,861✔
637
        frames = &frames_container->toclient;
20,211✔
638
    }
20,211✔
639

640
    SCLogDebug("flow %p direction %s frame type %u offset %" PRIu64 " len %" PRIi64
43,072✔
641
               " (slice offset %" PRIu64 ")",
43,072✔
642
            f, dir == 0 ? "toserver" : "toclient", frame_type, frame_start, len,
43,072✔
643
            stream_slice->offset);
43,072✔
644
    Frame *r = FrameNew(frames, frame_start, len);
43,072✔
645
    if (r != NULL) {
43,072✔
646
        r->type = frame_type;
43,072✔
647
    }
43,072✔
648
    return r;
43,072✔
649
}
43,072✔
650

651
void AppLayerFrameAddEvent(Frame *r, uint8_t e)
652
{
143✔
653
    if (r != NULL) {
143✔
654
        if (r->event_cnt < 4) { // TODO
143✔
655
            r->events[r->event_cnt++] = e;
143✔
656
        }
143✔
657
        FrameDebug("add_event", NULL, r);
143✔
658
    }
143✔
659
}
143✔
660

661
void SCAppLayerFrameAddEventById(const Flow *f, const int dir, const FrameId id, uint8_t e)
662
{
143✔
663
    Frame *frame = AppLayerFrameGetById(f, dir, id);
143✔
664
    AppLayerFrameAddEvent(frame, e);
143✔
665
}
143✔
666

667
void AppLayerFrameSetLength(Frame *frame, int64_t len)
668
{
45✔
669
    if (frame != NULL) {
45✔
670
        frame->len = len;
45✔
671
        FrameDebug("set_length", NULL, frame);
45✔
672
    }
45✔
673
}
45✔
674

675
void SCAppLayerFrameSetLengthById(const Flow *f, const int dir, const FrameId id, int64_t len)
676
{
45✔
677
    Frame *frame = AppLayerFrameGetById(f, dir, id);
45✔
678
    AppLayerFrameSetLength(frame, len);
45✔
679
}
45✔
680

681
void AppLayerFrameSetTxId(Frame *r, uint64_t tx_id)
682
{
160,353✔
683
    if (r != NULL) {
160,353✔
684
        r->flags |= FRAME_FLAG_TX_ID_SET;
160,353✔
685
        r->tx_id = tx_id;
160,353✔
686
        FrameDebug("set_txid", NULL, r);
160,353✔
687
    }
160,353✔
688
}
160,353✔
689

690
void SCAppLayerFrameSetTxIdById(const Flow *f, const int dir, const FrameId id, uint64_t tx_id)
691
{
124,753✔
692
    Frame *frame = AppLayerFrameGetById(f, dir, id);
124,753✔
693
    AppLayerFrameSetTxId(frame, tx_id);
124,753✔
694
}
124,753✔
695

696
Frame *AppLayerFrameGetById(const Flow *f, const int dir, const FrameId frame_id)
697
{
145,181✔
698
    FramesContainer *frames_container = AppLayerFramesGetContainer(f);
145,181✔
699
    SCLogDebug("get frame_id %" PRIi64 " direction %u/%s frames_container %p", frame_id, dir,
145,181✔
700
            dir == 0 ? "toserver" : "toclient", frames_container);
145,181✔
701
    if (frames_container == NULL)
145,181✔
702
        return NULL;
×
703

704
    Frames *frames;
145,181✔
705
    if (dir == 0) {
145,181✔
706
        frames = &frames_container->toserver;
86,247✔
707
    } else {
86,247✔
708
        frames = &frames_container->toclient;
58,934✔
709
    }
58,934✔
710
    SCLogDebug("frames %p", frames);
145,181✔
711
    return FrameGetById(frames, frame_id);
145,181✔
712
}
145,181✔
713

714
Frame *AppLayerFrameGetLastOpenByType(Flow *f, const int dir, const uint8_t frame_type)
715
{
352,451✔
716
    if (!(FrameConfigTypeIsEnabled(f->alproto, frame_type)))
352,451✔
717
        return NULL;
30,008✔
718

719
    FramesContainer *frames_container = AppLayerFramesGetContainer(f);
322,443✔
720
    SCLogDebug("get frame_type %" PRIu8 " direction %u/%s frames_container %p", frame_type, dir,
322,443✔
721
            dir == 0 ? "toserver" : "toclient", frames_container);
322,443✔
722
    if (frames_container == NULL)
322,443✔
723
        return NULL;
11,304✔
724

725
    Frames *frames;
311,139✔
726
    if (dir == 0) {
311,139✔
727
        frames = &frames_container->toserver;
291,654✔
728
    } else {
291,654✔
729
        frames = &frames_container->toclient;
19,485✔
730
    }
19,485✔
731
    SCLogDebug("frames %p", frames);
311,139✔
732
    return FrameGetLastOpenByType(frames, frame_type);
311,139✔
733
}
322,443✔
734

735
static inline bool FrameIsDone(const Frame *frame, const uint64_t abs_right_edge)
736
{
14,449,821✔
737
    /* frame with negative length means we don't know the size yet. */
738
    if (frame->len < 0)
14,449,821✔
739
        return false;
197,068✔
740

741
    const int64_t frame_abs_offset = frame->offset;
14,252,753✔
742
    const int64_t frame_right_edge = frame_abs_offset + frame->len;
14,252,753✔
743
    if ((uint64_t)frame_right_edge <= abs_right_edge) {
14,252,753✔
744
        SCLogDebug("frame %p id %" PRIi64 " is done", frame, frame->id);
899,619✔
745
        return true;
899,619✔
746
    }
899,619✔
747
    return false;
13,353,134✔
748
}
14,252,753✔
749

750
static void FramePrune(Frames *frames, const TcpStream *stream, const bool eof)
751
{
637,666✔
752
#ifdef DEBUG_VALIDATION
753
    const uint64_t frames_le_start = (uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream);
754
#endif
755
    SCLogDebug("start: left edge %" PRIu64 ", left_edge_rel %u, stream base %" PRIu64,
637,666✔
756
            (uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream), frames->left_edge_rel,
637,666✔
757
            STREAM_BASE_OFFSET(stream));
637,666✔
758
    const uint64_t acked = StreamTcpGetUsable(stream, eof);
637,666✔
759
    uint64_t le = STREAM_APP_PROGRESS(stream);
637,666✔
760

761
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
762
    const uint16_t start = frames->cnt;
763
    uint16_t removed = 0;
764
#endif
765
    uint16_t x = 0;
637,666✔
766
    for (uint16_t i = 0; i < frames->cnt; i++) {
15,141,331✔
767
        if (i < FRAMES_STATIC_CNT) {
14,503,665✔
768
            Frame *frame = &frames->sframes[i];
1,133,896✔
769
            FrameDebug("prune(s)", frames, frame);
1,133,896✔
770
            if (eof || FrameIsDone(frame, acked)) {
1,133,896✔
771
                // remove by not incrementing 'x'
772
                SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
662,644✔
773
                FrameDebug("remove(s)", frames, frame);
662,644✔
774
                FrameClean(frame);
662,644✔
775
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
776
                removed++;
777
#endif
778
            } else {
662,644✔
779
                const uint64_t fle = FrameLeftEdge(stream, frame);
471,252✔
780
                le = MIN(le, fle);
471,252✔
781
                SCLogDebug("le %" PRIu64 ", frame fle %" PRIu64, le, fle);
471,252✔
782
                Frame *nframe = &frames->sframes[x];
471,252✔
783
                FrameCopy(nframe, frame);
471,252✔
784
                if (frame != nframe) {
471,252✔
785
                    FrameClean(frame);
3,179✔
786
                }
3,179✔
787
                x++;
471,252✔
788
            }
471,252✔
789
        } else {
13,369,769✔
790
            const uint16_t o = i - FRAMES_STATIC_CNT;
13,369,769✔
791
            Frame *frame = &frames->dframes[o];
13,369,769✔
792
            FrameDebug("prune(d)", frames, frame);
13,369,769✔
793
            if (eof || FrameIsDone(frame, acked)) {
13,369,769✔
794
                // remove by not incrementing 'x'
795
                SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
290,819✔
796
                FrameDebug("remove(d)", frames, frame);
290,819✔
797
                FrameClean(frame);
290,819✔
798
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
799
                removed++;
800
#endif
801
            } else {
13,078,950✔
802
                const uint64_t fle = FrameLeftEdge(stream, frame);
13,078,950✔
803
                le = MIN(le, fle);
13,078,950✔
804
                SCLogDebug("le %" PRIu64 ", frame fle %" PRIu64, le, fle);
13,078,950✔
805
                Frame *nframe;
13,078,950✔
806
                if (x >= FRAMES_STATIC_CNT) {
13,078,950✔
807
                    nframe = &frames->dframes[x - FRAMES_STATIC_CNT];
13,072,335✔
808
                } else {
13,072,335✔
809
                    nframe = &frames->sframes[x];
6,615✔
810
                }
6,615✔
811
                FrameCopy(nframe, frame);
13,078,950✔
812
                if (frame != nframe) {
13,078,950✔
813
                    FrameClean(frame);
8,505✔
814
                }
8,505✔
815
                x++;
13,078,950✔
816
            }
13,078,950✔
817
        }
13,369,769✔
818
    }
14,503,665✔
819
    frames->cnt = x;
637,666✔
820
    DEBUG_VALIDATE_BUG_ON(le < STREAM_BASE_OFFSET(stream));
637,666✔
821
    DEBUG_VALIDATE_BUG_ON(le - STREAM_BASE_OFFSET(stream) > UINT32_MAX);
637,666✔
822
    frames->left_edge_rel = (uint32_t)(le - STREAM_BASE_OFFSET(stream));
637,666✔
823
#ifdef DEBUG
824
    SCLogDebug("end: left edge %" PRIu64 ", left_edge_rel %u, stream base %" PRIu64
825
               ", cnt %u, removed %u, start %u",
826
            (uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream), frames->left_edge_rel,
827
            STREAM_BASE_OFFSET(stream), frames->cnt, removed, start);
828
    AppLayerFrameDumpForFrames("post_slide", frames);
829
#endif
830
    if (frames->cnt > 0) { // if we removed all this can fail
637,666✔
831
        DEBUG_VALIDATE_BUG_ON(frames_le_start > le);
226,082✔
832
    }
226,082✔
833
    DEBUG_VALIDATE_BUG_ON(x != start - removed);
637,666✔
834
}
637,666✔
835

836
void FramesPrune(Flow *f, Packet *p)
837
{
766,296✔
838
    if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
766,296✔
839
        return;
×
840
    FramesContainer *frames_container = AppLayerFramesGetContainer(f);
766,296✔
841
    if (frames_container == NULL)
766,296✔
842
        return;
100,464✔
843

844
    Frames *frames;
665,832✔
845

846
    if (p->proto == IPPROTO_UDP) {
665,832✔
847
        SCLogDebug("clearing all UDP frames");
26,311✔
848
        if (PKT_IS_TOSERVER(p)) {
26,311✔
849
            frames = &frames_container->toserver;
20,972✔
850
        } else {
20,972✔
851
            frames = &frames_container->toclient;
5,339✔
852
        }
5,339✔
853
        FramesClear(frames);
26,311✔
854
        return;
26,311✔
855
    }
26,311✔
856

857
    TcpSession *ssn = f->protoctx;
639,521✔
858

859
    if (ssn->flags & STREAMTCP_FLAG_APP_LAYER_DISABLED) {
639,521✔
860
        AppLayerFramesFreeContainer(f);
1,855✔
861
        return;
1,855✔
862
    }
1,855✔
863

864
    TcpStream *stream;
637,666✔
865
    if (PKT_IS_TOSERVER(p)) {
637,666✔
866
        stream = &ssn->client;
321,568✔
867
        frames = &frames_container->toserver;
321,568✔
868
    } else {
321,568✔
869
        stream = &ssn->server;
316,098✔
870
        frames = &frames_container->toclient;
316,098✔
871
    }
316,098✔
872

873
    const bool eof = ssn->state == TCP_CLOSED || PKT_IS_PSEUDOPKT(p);
637,666✔
874
    SCLogDebug("eof %s", eof ? "TRUE" : "false");
637,666✔
875
    FramePrune(frames, stream, eof);
637,666✔
876
}
637,666✔
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc