diff options
Diffstat (limited to 'epan/stream.c')
-rw-r--r-- | epan/stream.c | 429 |
1 files changed, 429 insertions, 0 deletions
diff --git a/epan/stream.c b/epan/stream.c new file mode 100644 index 00000000..5bad6037 --- /dev/null +++ b/epan/stream.c @@ -0,0 +1,429 @@ +/* stream.c + * + * Definititions for handling circuit-switched protocols + * which are handled as streams, and don't have lengths + * and IDs such as are required for reassemble.h + * + * Wireshark - Network traffic analyzer + * By Gerald Combs <gerald@wireshark.org> + * Copyright 1998 Gerald Combs + * + * SPDX-License-Identifier: GPL-2.0-or-later + */ + +#include "config.h" + +#include <glib.h> +#include <epan/packet.h> +#include <epan/reassemble.h> +#include <epan/stream.h> +#include <epan/tvbuff.h> +#include <wsutil/ws_assert.h> + + +typedef struct { + fragment_head *fd_head; /* the reassembled data, NULL + * until we add the last fragment */ + guint32 pdu_number; /* Number of this PDU within the stream */ + + /* id of this pdu (globally unique) */ + guint32 id; +} stream_pdu_t; + + +struct stream_pdu_fragment +{ + guint32 len; /* the length of this fragment */ + stream_pdu_t *pdu; + gboolean final_fragment; +}; + +struct stream { + /* the key used to add this stream to stream_hash */ + struct stream_key *key; + + /* pdu to add the next fragment to, or NULL if we need to start + * a new PDU. + */ + stream_pdu_t *current_pdu; + + /* number of PDUs added to this stream so far */ + guint32 pdu_counter; + + /* the framenumber and offset of the last fragment added; + used for sanity-checking */ + guint32 lastfrag_framenum; + guint32 lastfrag_offset; +}; + + +/***************************************************************************** + * + * Stream hash + */ + +/* key */ +typedef struct stream_key { + /* streams are attached to conversations */ + const struct conversation *conv; + int p2p_dir; +} stream_key_t; + + +/* hash func */ +static guint stream_hash_func(gconstpointer k) +{ + const stream_key_t *key = (const stream_key_t *)k; + + return (GPOINTER_TO_UINT(key->conv)) ^ key->p2p_dir; +} + +/* compare func */ +static gboolean stream_compare_func(gconstpointer a, + gconstpointer b) +{ + const stream_key_t *key1 = (const stream_key_t *)a; + const stream_key_t *key2 = (const stream_key_t *)b; + if( key1 -> p2p_dir != key2 -> p2p_dir) + return FALSE; + + return (key1 -> conv == key2 -> conv ); +} + +/* the hash table */ +static GHashTable *stream_hash; + + +/* cleanup reset function, call from stream_cleanup() */ +static void cleanup_stream_hash( void ) { + if( stream_hash != NULL ) { + g_hash_table_destroy( stream_hash ); + stream_hash = NULL; + } +} + +/* init function, call from stream_init() */ +static void init_stream_hash( void ) { + ws_assert(stream_hash==NULL); + stream_hash = g_hash_table_new(stream_hash_func, + stream_compare_func); +} + +/* lookup function, returns null if not found */ +static stream_t *stream_hash_lookup( const struct conversation *conv, int p2p_dir ) +{ + stream_key_t key; + key.conv = conv; + key.p2p_dir=p2p_dir; + return (stream_t *)g_hash_table_lookup(stream_hash, &key); +} + + +static stream_t *new_stream( stream_key_t *key ) +{ + stream_t *val; + + val = wmem_new(wmem_file_scope(), stream_t); + val -> key = key; + val -> pdu_counter = 0; + val -> current_pdu = NULL; + val -> lastfrag_framenum = 0; + val -> lastfrag_offset = 0; + g_hash_table_insert(stream_hash, key, val); + + return val; +} + + +/* insert function */ +static stream_t *stream_hash_insert( const struct conversation *conv, int p2p_dir ) +{ + stream_key_t *key; + + key = wmem_new(wmem_file_scope(), stream_key_t); + key->conv = conv; + key->p2p_dir = p2p_dir; + + return new_stream(key); +} + + +/****************************************************************************** + * + * PDU data + */ + +/* pdu counter, for generating unique pdu ids */ +static guint32 pdu_counter; + +static void stream_cleanup_pdu_data(void) +{ +} + +static void stream_init_pdu_data(void) +{ + pdu_counter = 0; +} + + +/* new pdu in this stream */ +static stream_pdu_t *stream_new_pdu(stream_t *stream) +{ + stream_pdu_t *pdu; + pdu = wmem_new(wmem_file_scope(), stream_pdu_t); + pdu -> fd_head = NULL; + pdu -> pdu_number = stream -> pdu_counter++; + pdu -> id = pdu_counter++; + return pdu; +} + +/***************************************************************************** + * + * fragment hash + */ + +/* key */ +typedef struct fragment_key { + const stream_t *stream; + guint32 framenum; + guint32 offset; +} fragment_key_t; + + +/* hash func */ +static guint fragment_hash_func(gconstpointer k) +{ + const fragment_key_t *key = (const fragment_key_t *)k; + return (GPOINTER_TO_UINT(key->stream)) + ((guint)key -> framenum) + ((guint)key->offset); +} + +/* compare func */ +static gboolean fragment_compare_func(gconstpointer a, + gconstpointer b) +{ + const fragment_key_t *key1 = (const fragment_key_t *)a; + const fragment_key_t *key2 = (const fragment_key_t *)b; + return (key1 -> stream == key2 -> stream && + key1 -> framenum == key2 -> framenum && + key1 -> offset == key2 -> offset ); +} + +/* the hash table */ +static GHashTable *fragment_hash; + + +/* cleanup function, call from stream_cleanup() */ +static void cleanup_fragment_hash( void ) { + if( fragment_hash != NULL ) { + g_hash_table_destroy( fragment_hash ); + fragment_hash = NULL; + } +} + +/* init function, call from stream_init() */ +static void init_fragment_hash( void ) { + ws_assert(fragment_hash==NULL); + fragment_hash = g_hash_table_new(fragment_hash_func, + fragment_compare_func); +} + + +/* lookup function, returns null if not found */ +static stream_pdu_fragment_t *fragment_hash_lookup( const stream_t *stream, guint32 framenum, guint32 offset ) +{ + fragment_key_t key; + stream_pdu_fragment_t *val; + + key.stream = stream; + key.framenum = framenum; + key.offset = offset; + val = (stream_pdu_fragment_t *)g_hash_table_lookup(fragment_hash, &key); + + return val; +} + + +/* insert function */ +static stream_pdu_fragment_t *fragment_hash_insert( const stream_t *stream, guint32 framenum, guint32 offset, + guint32 length) +{ + fragment_key_t *key; + stream_pdu_fragment_t *val; + + key = wmem_new(wmem_file_scope(), fragment_key_t); + key->stream = stream; + key->framenum = framenum; + key->offset = offset; + + val = wmem_new(wmem_file_scope(), stream_pdu_fragment_t); + val->len = length; + val->pdu = NULL; + val->final_fragment = FALSE; + + g_hash_table_insert(fragment_hash, key, val); + return val; +} + +/*****************************************************************************/ + +/* reassembly table */ +static reassembly_table stream_reassembly_table; + +/* Initialise a new stream. Call this when you first identify a distinct + * stream. */ +stream_t *stream_new ( const struct conversation *conv, int p2p_dir ) +{ + stream_t * stream; + + /* we don't want to replace the previous data if we get called twice on the + same conversation, so do a lookup first */ + stream = stream_hash_lookup(conv, p2p_dir); + DISSECTOR_ASSERT( stream == NULL ); + + stream = stream_hash_insert(conv, p2p_dir); + return stream; +} + + +/* retrieve a previously-created stream. + * + * Returns null if no matching stream was found. + */ +stream_t *find_stream ( const struct conversation *conv, int p2p_dir ) +{ + return stream_hash_lookup(conv,p2p_dir); +} + +/* cleanup the stream routines */ +/* Note: stream_cleanup must only be called when seasonal memory + * is also freed since the hash tables countain pointers to + * wmem_file_scoped memory. + */ +void stream_cleanup( void ) +{ + cleanup_stream_hash(); + cleanup_fragment_hash(); + stream_cleanup_pdu_data(); + reassembly_table_destroy(&stream_reassembly_table); +} + +/* initialise the stream routines */ +void stream_init( void ) +{ + init_stream_hash(); + init_fragment_hash(); + stream_init_pdu_data(); + + reassembly_table_init(&stream_reassembly_table, + &addresses_reassembly_table_functions); +} + +/*****************************************************************************/ + +stream_pdu_fragment_t *stream_find_frag( stream_t *stream, guint32 framenum, guint32 offset ) +{ + return fragment_hash_lookup( stream, framenum, offset ); +} + +stream_pdu_fragment_t *stream_add_frag( stream_t *stream, guint32 framenum, guint32 offset, + tvbuff_t *tvb, packet_info *pinfo, gboolean more_frags ) +{ + fragment_head *fd_head; + stream_pdu_t *pdu; + stream_pdu_fragment_t *frag_data; + + DISSECTOR_ASSERT(stream); + + /* check that this fragment is at the end of the stream */ + DISSECTOR_ASSERT( framenum > stream->lastfrag_framenum || + (framenum == stream->lastfrag_framenum && offset > stream->lastfrag_offset)); + + + pdu = stream->current_pdu; + if( pdu == NULL ) { + /* start a new pdu */ + pdu = stream->current_pdu = stream_new_pdu(stream); + } + + /* add it to the reassembly tables */ + fd_head = fragment_add_seq_next(&stream_reassembly_table, + tvb, 0, pinfo, pdu->id, NULL, + tvb_reported_length(tvb), more_frags); + /* add it to our hash */ + frag_data = fragment_hash_insert( stream, framenum, offset, tvb_reported_length(tvb)); + frag_data -> pdu = pdu; + + if( fd_head != NULL ) { + /* if this was the last fragment, update the pdu data. + */ + pdu -> fd_head = fd_head; + + /* start a new pdu next time */ + stream->current_pdu = NULL; + + frag_data -> final_fragment = TRUE; + } + + /* stashing the framenum and offset permit future sanity checks */ + stream -> lastfrag_framenum = framenum; + stream -> lastfrag_offset = offset; + + return frag_data; +} + + +tvbuff_t *stream_process_reassembled( + tvbuff_t *tvb, int offset, packet_info *pinfo, + const char *name, const stream_pdu_fragment_t *frag, + const struct _fragment_items *fit, + gboolean *update_col_infop, proto_tree *tree) +{ + stream_pdu_t *pdu; + DISSECTOR_ASSERT(frag); + pdu = frag->pdu; + + /* we handle non-terminal fragments ourselves, because + reassemble.c messes them up */ + if(!frag->final_fragment) { + if (pdu->fd_head != NULL && fit->hf_reassembled_in != NULL) { + proto_tree_add_uint(tree, + *(fit->hf_reassembled_in), tvb, + 0, 0, pdu->fd_head->reassembled_in); + } + return NULL; + } + + return process_reassembled_data(tvb, offset, pinfo, name, pdu->fd_head, + fit, update_col_infop, tree); +} + +guint32 stream_get_frag_length( const stream_pdu_fragment_t *frag) +{ + DISSECTOR_ASSERT( frag ); + return frag->len; +} + +fragment_head *stream_get_frag_data( const stream_pdu_fragment_t *frag) +{ + DISSECTOR_ASSERT( frag ); + return frag->pdu->fd_head; +} + +guint32 stream_get_pdu_no( const stream_pdu_fragment_t *frag) +{ + DISSECTOR_ASSERT( frag ); + return frag->pdu->pdu_number; +} + +/* + * Editor modelines - https://www.wireshark.org/tools/modelines.html + * + * Local variables: + * c-basic-offset: 4 + * tab-width: 8 + * indent-tabs-mode: nil + * End: + * + * vi: set shiftwidth=4 tabstop=8 expandtab: + * :indentSize=4:tabSize=8:noTabs=true: + */ |