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

OISF / suricata / 22553697083

01 Mar 2026 09:58PM UTC coverage: 75.673% (+2.0%) from 73.687%
22553697083

Pull #14925

github

web-flow
Merge 288827f07 into 90823fa90
Pull Request #14925: hs: false positive coverity warning in a reference string v1

241615 of 319288 relevant lines covered (75.67%)

1333554.73 hits per line

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

96.22
/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
{
37✔
42
    frame_config = SCCalloc(g_alproto_max, sizeof(struct FrameConfig));
37✔
43
    if (unlikely(frame_config == NULL)) {
37✔
44
        FatalError("Unable to alloc frame_config.");
×
45
    }
×
46
    for (AppProto p = 0; p < g_alproto_max; p++) {
1,517✔
47
        SC_ATOMIC_INIT(frame_config[p].types);
1,480✔
48
    }
1,480✔
49
}
37✔
50

51
void FrameConfigDeInit(void)
52
{
23✔
53
    SCFree(frame_config);
23✔
54
}
23✔
55

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

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

72
static inline bool FrameConfigTypeIsEnabled(const AppProto p, const uint8_t type)
73
{
2,801,420✔
74
    struct FrameConfig *fc = &frame_config[p];
2,801,420✔
75
    const uint64_t bits = BIT_U64(type);
2,801,420✔
76
    const bool enabled = (SC_ATOMIC_GET(fc->types) & bits) != 0;
2,801,420✔
77
    return enabled;
2,801,420✔
78
}
2,801,420✔
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,333✔
104
    Frame *candidate = NULL;
311,333✔
105

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

125
Frame *FrameGetById(Frames *frames, const int64_t id)
126
{
149,166✔
127
    SCLogDebug("frames %p cnt %u, looking for %" PRIi64, frames, frames->cnt, id);
149,166✔
128
    for (uint16_t i = 0; i < frames->cnt; i++) {
1,970,345✔
129
        if (i < FRAMES_STATIC_CNT) {
1,970,235✔
130
            Frame *frame = &frames->sframes[i];
367,444✔
131
            FrameDebug("get_by_id(static)", frames, frame);
367,444✔
132
            if (frame->id == id)
367,444✔
133
                return frame;
68,378✔
134
        } else {
1,602,791✔
135
            const uint16_t o = i - FRAMES_STATIC_CNT;
1,602,791✔
136
            Frame *frame = &frames->dframes[o];
1,602,791✔
137
            FrameDebug("get_by_id(dynamic)", frames, frame);
1,602,791✔
138
            if (frame->id == id)
1,602,791✔
139
                return frame;
80,678✔
140
        }
1,602,791✔
141
    }
1,970,235✔
142
    return NULL;
110✔
143
}
149,166✔
144

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

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

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

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

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

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

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

197
            // fall through
198
        } else {
403,092✔
199
            if (frames->dyn_size == 256) {
109,563✔
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,466✔
208
            uint32_t new_alloc_size = new_dyn_size * sizeof(Frame);
7,466✔
209

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

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

221
        frames->cnt++;
410,558✔
222
        frames->dframes[dyn_cnt].offset = offset;
410,558✔
223
        frames->dframes[dyn_cnt].len = len;
410,558✔
224
        frames->dframes[dyn_cnt].id = ++frames->base_id;
410,558✔
225
        return &frames->dframes[dyn_cnt];
410,558✔
226
    }
512,655✔
227
}
1,245,129✔
228

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

234
static void FrameCopy(Frame *dst, Frame *src)
235
{
14,004,766✔
236
    memcpy(dst, src, sizeof(*dst));
14,004,766✔
237
}
14,004,766✔
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,445,225✔
259
    const int64_t app_progress = STREAM_APP_PROGRESS(stream);
14,445,225✔
260

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

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

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

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

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

292
             */
293
        } else if (frame_data < 2500) {
14,216,594✔
294
            uint64_t x = frame_offset;
14,004✔
295
            SCLogDebug("x %" PRIu64, x);
14,004✔
296
            return x;
14,004✔
297
        } else {
14,004✔
298
            uint64_t x = frame_offset + (frame_data - 2500);
11,217✔
299
            SCLogDebug("x %" PRIu64, x);
11,217✔
300
            return x;
11,217✔
301
        }
11,217✔
302
    }
14,241,815✔
303
}
14,445,225✔
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,832✔
330
    SCLogDebug("start: left edge %" PRIu64 ", left_edge_rel %u, stream base %" PRIu64
15,832✔
331
               ", next %" PRIu64,
15,832✔
332
            (uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream), frames->left_edge_rel,
15,832✔
333
            STREAM_BASE_OFFSET(stream), STREAM_BASE_OFFSET(stream) + slide);
15,832✔
334
    DEBUG_VALIDATE_BUG_ON(frames == NULL);
15,832✔
335
    SCLogDebug("%s frames %p: sliding %u bytes", ds, frames, slide);
15,832✔
336
    uint64_t le = STREAM_APP_PROGRESS(stream);
15,832✔
337
    const uint64_t next_base = STREAM_BASE_OFFSET(stream) + slide;
15,832✔
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,832✔
343
    for (uint16_t i = 0; i < frames->cnt; i++) {
462,545✔
344
        if (i < FRAMES_STATIC_CNT) {
446,713✔
345
            Frame *frame = &frames->sframes[i];
11,381✔
346
            FrameDebug("slide(s)", frames, frame);
11,381✔
347
            if (frame->len >= 0 && frame->offset + frame->len <= next_base) {
11,381✔
348
                // remove by not incrementing 'x'
349
                SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
192✔
350
                FrameClean(frame);
192✔
351
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
352
                removed++;
353
#endif
354
            } else {
11,189✔
355
                Frame *nframe = &frames->sframes[x];
11,189✔
356
                FrameCopy(nframe, frame);
11,189✔
357
                if (frame != nframe) {
11,189✔
358
                    FrameClean(frame);
10✔
359
                }
10✔
360
                le = MIN(le, FrameLeftEdge(stream, nframe));
11,189✔
361
                x++;
11,189✔
362
            }
11,189✔
363
        } else {
435,332✔
364
            const uint16_t o = i - FRAMES_STATIC_CNT;
435,332✔
365
            Frame *frame = &frames->dframes[o];
435,332✔
366
            FrameDebug("slide(d)", frames, frame);
435,332✔
367
            if (frame->len >= 0 && frame->offset + frame->len <= next_base) {
435,332✔
368
                // remove by not incrementing 'x'
369
                SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
5,870✔
370
                FrameClean(frame);
5,870✔
371
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
372
                removed++;
373
#endif
374
            } else {
429,462✔
375
                Frame *nframe;
429,462✔
376
                if (x >= FRAMES_STATIC_CNT) {
429,462✔
377
                    nframe = &frames->dframes[x - FRAMES_STATIC_CNT];
429,298✔
378
                } else {
429,298✔
379
                    nframe = &frames->sframes[x];
164✔
380
                }
164✔
381
                FrameCopy(nframe, frame);
429,462✔
382
                if (frame != nframe) {
429,462✔
383
                    FrameClean(frame);
6,541✔
384
                }
6,541✔
385
                le = MIN(le, FrameLeftEdge(stream, nframe));
429,462✔
386
                x++;
429,462✔
387
            }
429,462✔
388
        }
435,332✔
389
    }
446,713✔
390
    frames->cnt = x;
15,832✔
391
    uint64_t o = STREAM_BASE_OFFSET(stream) + slide;
15,832✔
392
    DEBUG_VALIDATE_BUG_ON(o > le);
15,832✔
393
    DEBUG_VALIDATE_BUG_ON(le - o > UINT32_MAX);
15,832✔
394
    frames->left_edge_rel = (uint32_t)(le - o);
15,832✔
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,832✔
406
    return 0;
15,832✔
407
}
15,832✔
408

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

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

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

438
    SCLogDebug("frames %u", frames->cnt);
82,042✔
439
    for (uint16_t i = 0; i < frames->cnt; i++) {
262,824✔
440
        if (i < FRAMES_STATIC_CNT) {
180,782✔
441
            Frame *r = &frames->sframes[i];
66,214✔
442
            SCLogDebug("removing frame %p", r);
66,214✔
443
            FrameFreeSingleFrame(frames, r);
66,214✔
444
        } else {
114,568✔
445
            const uint16_t o = i - FRAMES_STATIC_CNT;
114,568✔
446
            Frame *r = &frames->dframes[o];
114,568✔
447
            SCLogDebug("removing frame %p", r);
114,568✔
448
            FrameFreeSingleFrame(frames, r);
114,568✔
449
        }
114,568✔
450
    }
180,782✔
451
    frames->cnt = 0;
82,042✔
452
}
82,042✔
453

454
void FramesFree(Frames *frames)
455
{
55,398✔
456
    DEBUG_VALIDATE_BUG_ON(frames == NULL);
55,398✔
457
    FramesClear(frames);
55,398✔
458
    SCFree(frames->dframes);
55,398✔
459
    frames->dframes = NULL;
55,398✔
460
}
55,398✔
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,920✔
467
    SCLogDebug("frame_start:%p stream_slice->input:%p stream_slice->offset:%" PRIu64, frame_start,
126,920✔
468
            stream_slice->input, stream_slice->offset);
126,920✔
469

470
    if (!(FrameConfigTypeIsEnabled(f->alproto, frame_type)))
126,920✔
471
        return NULL;
73,845✔
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,075✔
475
    if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
53,075✔
476
        return NULL;
24✔
477
    if (frame_start < stream_slice->input ||
53,051✔
478
            frame_start > stream_slice->input + stream_slice->input_len)
53,051✔
479
        return NULL;
480
#endif
53,051✔
481
    DEBUG_VALIDATE_BUG_ON(frame_start < stream_slice->input);
53,051✔
482
    DEBUG_VALIDATE_BUG_ON(stream_slice->input == NULL);
53,051✔
483
    DEBUG_VALIDATE_BUG_ON(f->proto == IPPROTO_TCP && f->protoctx == NULL);
53,051✔
484

485
    ptrdiff_t ptr_offset = frame_start - stream_slice->input;
53,051✔
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,051✔
493

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

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

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

507
    Frame *r = FrameNew(frames, abs_frame_offset, len);
53,051✔
508
    if (r != NULL) {
53,051✔
509
        r->type = frame_type;
52,045✔
510
        FrameDebug("new_by_ptr", frames, r);
52,045✔
511
    }
52,045✔
512
    return r;
53,051✔
513
}
53,051✔
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
{
32,113✔
518
    DEBUG_VALIDATE_BUG_ON(f->proto != IPPROTO_UDP);
32,113✔
519

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

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

527
    Frames *frames;
32,113✔
528
    if (dir == 0) {
32,113✔
529
        frames = &frames_container->toserver;
25,645✔
530
    } else {
25,645✔
531
        frames = &frames_container->toclient;
6,468✔
532
    }
6,468✔
533

534
    Frame *r = FrameNew(frames, frame_start_rel, len);
32,113✔
535
    if (r != NULL) {
32,113✔
536
        r->type = frame_type;
32,113✔
537
    }
32,113✔
538
    return r;
32,113✔
539
}
32,113✔
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,221,543✔
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,221,543✔
550
    if (!(FrameConfigTypeIsEnabled(f->alproto, frame_type)))
2,221,543✔
551
        return NULL;
1,072,607✔
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,148,936✔
555
    if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
1,148,936✔
556
        return NULL;
557
    if (stream_slice->input == NULL)
1,148,936✔
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,148,936✔
563
    DEBUG_VALIDATE_BUG_ON(f->alparser == NULL);
1,148,936✔
564

565
    if (f->proto == IPPROTO_UDP) {
1,148,936✔
566
        return AppLayerFrameUdp(f, frame_start_rel, len, dir, frame_type);
32,113✔
567
    }
32,113✔
568

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

573
    Frames *frames;
1,116,823✔
574
    if (dir == 0) {
1,116,823✔
575
        frames = &frames_container->toserver;
555,928✔
576
    } else {
560,895✔
577
        frames = &frames_container->toclient;
560,895✔
578
    }
560,895✔
579

580
    const uint64_t frame_abs_offset = (uint64_t)frame_start_rel + stream_slice->offset;
1,116,823✔
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,116,823✔
588
    if (r != NULL) {
1,116,823✔
589
        r->type = frame_type;
1,015,732✔
590
    }
1,015,732✔
591
    return r;
1,116,823✔
592
}
1,116,823✔
593

594
void AppLayerFrameDump(Flow *f)
595
{
497,928✔
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
}
497,928✔
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
{
67,249✔
612
    if (!(FrameConfigTypeIsEnabled(f->alproto, frame_type)))
67,249✔
613
        return NULL;
24,106✔
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,143✔
617
    if (f->proto == IPPROTO_TCP && f->protoctx == NULL)
43,143✔
618
        return NULL;
619
    if (stream_slice->input == NULL)
43,143✔
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,142✔
625
    DEBUG_VALIDATE_BUG_ON(f->alparser == NULL);
43,142✔
626
    DEBUG_VALIDATE_BUG_ON(frame_start < stream_slice->offset);
43,142✔
627
    DEBUG_VALIDATE_BUG_ON(frame_start - stream_slice->offset >= (uint64_t)INT_MAX);
43,142✔
628

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

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

640
    SCLogDebug("flow %p direction %s frame type %u offset %" PRIu64 " len %" PRIi64
43,142✔
641
               " (slice offset %" PRIu64 ")",
43,142✔
642
            f, dir == 0 ? "toserver" : "toclient", frame_type, frame_start, len,
43,142✔
643
            stream_slice->offset);
43,142✔
644
    Frame *r = FrameNew(frames, frame_start, len);
43,142✔
645
    if (r != NULL) {
43,142✔
646
        r->type = frame_type;
43,142✔
647
    }
43,142✔
648
    return r;
43,142✔
649
}
43,142✔
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
{
162,654✔
683
    if (r != NULL) {
162,654✔
684
        r->flags |= FRAME_FLAG_TX_ID_SET;
162,654✔
685
        r->tx_id = tx_id;
162,654✔
686
        FrameDebug("set_txid", NULL, r);
162,654✔
687
    }
162,654✔
688
}
162,654✔
689

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

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

704
    Frames *frames;
147,446✔
705
    if (dir == 0) {
147,446✔
706
        frames = &frames_container->toserver;
87,556✔
707
    } else {
87,556✔
708
        frames = &frames_container->toclient;
59,890✔
709
    }
59,890✔
710
    SCLogDebug("frames %p", frames);
147,446✔
711
    return FrameGetById(frames, frame_id);
147,446✔
712
}
147,446✔
713

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

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

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

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

741
    const int64_t frame_abs_offset = frame->offset;
14,269,236✔
742
    const int64_t frame_right_edge = frame_abs_offset + frame->len;
14,269,236✔
743
    if ((uint64_t)frame_right_edge <= abs_right_edge) {
14,269,236✔
744
        SCLogDebug("frame %p id %" PRIi64 " is done", frame, frame->id);
902,326✔
745
        return true;
902,326✔
746
    }
902,326✔
747
    return false;
13,366,910✔
748
}
14,269,236✔
749

750
static void FramePrune(Frames *frames, const TcpStream *stream, const bool eof)
751
{
638,470✔
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,
638,470✔
756
            (uint64_t)frames->left_edge_rel + STREAM_BASE_OFFSET(stream), frames->left_edge_rel,
638,470✔
757
            STREAM_BASE_OFFSET(stream));
638,470✔
758
    const uint64_t acked = StreamTcpGetUsable(stream, eof);
638,470✔
759
    uint64_t le = STREAM_APP_PROGRESS(stream);
638,470✔
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;
638,470✔
766
    for (uint16_t i = 0; i < frames->cnt; i++) {
15,158,773✔
767
        if (i < FRAMES_STATIC_CNT) {
14,520,303✔
768
            Frame *frame = &frames->sframes[i];
1,136,059✔
769
            FrameDebug("prune(s)", frames, frame);
1,136,059✔
770
            if (eof || FrameIsDone(frame, acked)) {
1,136,059✔
771
                // remove by not incrementing 'x'
772
                SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
664,297✔
773
                FrameDebug("remove(s)", frames, frame);
664,297✔
774
                FrameClean(frame);
664,297✔
775
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
776
                removed++;
777
#endif
778
            } else {
664,297✔
779
                const uint64_t fle = FrameLeftEdge(stream, frame);
471,762✔
780
                le = MIN(le, fle);
471,762✔
781
                SCLogDebug("le %" PRIu64 ", frame fle %" PRIu64, le, fle);
471,762✔
782
                Frame *nframe = &frames->sframes[x];
471,762✔
783
                FrameCopy(nframe, frame);
471,762✔
784
                if (frame != nframe) {
471,762✔
785
                    FrameClean(frame);
3,177✔
786
                }
3,177✔
787
                x++;
471,762✔
788
            }
471,762✔
789
        } else {
13,384,244✔
790
            const uint16_t o = i - FRAMES_STATIC_CNT;
13,384,244✔
791
            Frame *frame = &frames->dframes[o];
13,384,244✔
792
            FrameDebug("prune(d)", frames, frame);
13,384,244✔
793
            if (eof || FrameIsDone(frame, acked)) {
13,384,244✔
794
                // remove by not incrementing 'x'
795
                SCLogDebug("removing %p id %" PRIi64, frame, frame->id);
291,891✔
796
                FrameDebug("remove(d)", frames, frame);
291,891✔
797
                FrameClean(frame);
291,891✔
798
#if defined(DEBUG) || defined(DEBUG_VALIDATION)
799
                removed++;
800
#endif
801
            } else {
13,092,353✔
802
                const uint64_t fle = FrameLeftEdge(stream, frame);
13,092,353✔
803
                le = MIN(le, fle);
13,092,353✔
804
                SCLogDebug("le %" PRIu64 ", frame fle %" PRIu64, le, fle);
13,092,353✔
805
                Frame *nframe;
13,092,353✔
806
                if (x >= FRAMES_STATIC_CNT) {
13,092,353✔
807
                    nframe = &frames->dframes[x - FRAMES_STATIC_CNT];
13,085,473✔
808
                } else {
13,085,473✔
809
                    nframe = &frames->sframes[x];
6,880✔
810
                }
6,880✔
811
                FrameCopy(nframe, frame);
13,092,353✔
812
                if (frame != nframe) {
13,092,353✔
813
                    FrameClean(frame);
8,770✔
814
                }
8,770✔
815
                x++;
13,092,353✔
816
            }
13,092,353✔
817
        }
13,384,244✔
818
    }
14,520,303✔
819
    frames->cnt = x;
638,470✔
820
    DEBUG_VALIDATE_BUG_ON(le < STREAM_BASE_OFFSET(stream));
638,470✔
821
    DEBUG_VALIDATE_BUG_ON(le - STREAM_BASE_OFFSET(stream) > UINT32_MAX);
638,470✔
822
    frames->left_edge_rel = (uint32_t)(le - STREAM_BASE_OFFSET(stream));
638,470✔
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
638,470✔
831
        DEBUG_VALIDATE_BUG_ON(frames_le_start > le);
226,638✔
832
    }
226,638✔
833
    DEBUG_VALIDATE_BUG_ON(x != start - removed);
638,470✔
834
}
638,470✔
835

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

844
    Frames *frames;
666,970✔
845

846
    if (p->proto == IPPROTO_UDP) {
666,970✔
847
        SCLogDebug("clearing all UDP frames");
26,644✔
848
        if (PKT_IS_TOSERVER(p)) {
26,644✔
849
            frames = &frames_container->toserver;
21,313✔
850
        } else {
21,313✔
851
            frames = &frames_container->toclient;
5,331✔
852
        }
5,331✔
853
        FramesClear(frames);
26,644✔
854
        return;
26,644✔
855
    }
26,644✔
856

857
    TcpSession *ssn = f->protoctx;
640,326✔
858

859
    if (ssn->flags & STREAMTCP_FLAG_APP_LAYER_DISABLED) {
640,326✔
860
        AppLayerFramesFreeContainer(f);
1,856✔
861
        return;
1,856✔
862
    }
1,856✔
863

864
    TcpStream *stream;
638,470✔
865
    if (PKT_IS_TOSERVER(p)) {
638,470✔
866
        stream = &ssn->client;
322,013✔
867
        frames = &frames_container->toserver;
322,013✔
868
    } else {
322,013✔
869
        stream = &ssn->server;
316,457✔
870
        frames = &frames_container->toclient;
316,457✔
871
    }
316,457✔
872

873
    const bool eof = ssn->state == TCP_CLOSED || PKT_IS_PSEUDOPKT(p);
638,470✔
874
    SCLogDebug("eof %s", eof ? "TRUE" : "false");
638,470✔
875
    FramePrune(frames, stream, eof);
638,470✔
876
}
638,470✔
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