2011-04-27 02:54:44 +00:00
|
|
|
/* frame_data_sequence.c
|
|
|
|
* Implements a sequence of frame_data structures
|
|
|
|
*
|
|
|
|
* Wireshark - Network traffic analyzer
|
|
|
|
* By Gerald Combs <gerald@wireshark.org>
|
|
|
|
* Copyright 1998 Gerald Combs
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
2012-06-28 22:56:06 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
2011-04-27 02:54:44 +00:00
|
|
|
*/
|
|
|
|
|
2012-09-20 01:29:52 +00:00
|
|
|
#include "config.h"
|
2011-04-27 02:54:44 +00:00
|
|
|
|
|
|
|
#include <glib.h>
|
|
|
|
|
|
|
|
#include <epan/packet.h>
|
|
|
|
|
|
|
|
#include "frame_data_sequence.h"
|
|
|
|
|
2011-04-27 04:11:37 +00:00
|
|
|
/*
|
|
|
|
* We store the frame_data structures in a radix tree, with 1024
|
|
|
|
* elements per level. The leaf nodes are arrays of 1024 frame_data
|
|
|
|
* structures; the nodes above them are arrays of 1024 pointers to
|
|
|
|
* the nodes below them. The capture_file structure has a pointer
|
|
|
|
* to the root node.
|
|
|
|
*
|
|
|
|
* As frame numbers are 32 bits, and as 1024 is 2^10, that gives us
|
|
|
|
* up to 4 levels of tree.
|
|
|
|
*/
|
2014-08-05 20:10:44 +00:00
|
|
|
#define LOG2_NODES_PER_LEVEL 10
|
|
|
|
#define NODES_PER_LEVEL (1<<LOG2_NODES_PER_LEVEL)
|
2011-04-27 02:54:44 +00:00
|
|
|
|
2011-04-27 04:11:37 +00:00
|
|
|
struct _frame_data_sequence {
|
|
|
|
guint32 count; /* Total number of frames */
|
|
|
|
void *ptree_root; /* Pointer to the root node */
|
|
|
|
};
|
2011-04-27 02:54:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* For a given frame number, calculate the indices into a level 3
|
|
|
|
* node, a level 2 node, a level 1 node, and a leaf node.
|
|
|
|
*/
|
|
|
|
#define LEVEL_3_INDEX(framenum) \
|
2014-08-05 20:10:44 +00:00
|
|
|
((framenum) >> (3*LOG2_NODES_PER_LEVEL))
|
2011-04-27 02:54:44 +00:00
|
|
|
#define LEVEL_2_INDEX(framenum) \
|
2014-08-05 20:10:44 +00:00
|
|
|
(((framenum) >> (2*LOG2_NODES_PER_LEVEL)) & (NODES_PER_LEVEL - 1))
|
2011-04-27 02:54:44 +00:00
|
|
|
#define LEVEL_1_INDEX(framenum) \
|
2014-08-05 20:10:44 +00:00
|
|
|
(((framenum) >> (1*LOG2_NODES_PER_LEVEL)) & (NODES_PER_LEVEL - 1))
|
2011-04-27 02:54:44 +00:00
|
|
|
#define LEAF_INDEX(framenum) \
|
2014-08-05 20:10:44 +00:00
|
|
|
(((framenum) >> (0*LOG2_NODES_PER_LEVEL)) & (NODES_PER_LEVEL - 1))
|
2011-04-27 02:54:44 +00:00
|
|
|
|
2011-04-27 04:11:37 +00:00
|
|
|
frame_data_sequence *
|
|
|
|
new_frame_data_sequence(void)
|
|
|
|
{
|
2014-08-05 20:10:44 +00:00
|
|
|
frame_data_sequence *fds;
|
2011-04-27 04:11:37 +00:00
|
|
|
|
2014-08-05 20:10:44 +00:00
|
|
|
fds = (frame_data_sequence *)g_malloc(sizeof *fds);
|
|
|
|
fds->count = 0;
|
|
|
|
fds->ptree_root = NULL;
|
|
|
|
return fds;
|
2011-04-27 04:11:37 +00:00
|
|
|
}
|
|
|
|
|
2011-04-27 02:54:44 +00:00
|
|
|
/*
|
|
|
|
* Add a new frame_data structure to a frame_data_sequence.
|
|
|
|
*/
|
|
|
|
frame_data *
|
|
|
|
frame_data_sequence_add(frame_data_sequence *fds, frame_data *fdata)
|
|
|
|
{
|
|
|
|
frame_data *leaf;
|
|
|
|
frame_data **level1;
|
|
|
|
frame_data ***level2;
|
|
|
|
frame_data ****level3;
|
|
|
|
frame_data *node;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The current value of fds->count is the index value for the new frame,
|
|
|
|
* because the index value for a frame is the frame number - 1, and
|
|
|
|
* if we currently have fds->count frames, the the frame number of
|
|
|
|
* the last frame in the collection is fds->count, so its index value
|
|
|
|
* is fds->count - 1.
|
|
|
|
*/
|
|
|
|
if (fds->count == 0) {
|
|
|
|
/* The tree is empty; allocate the first leaf node, which will be
|
|
|
|
the root node. */
|
2013-03-02 09:06:03 +00:00
|
|
|
leaf = (frame_data *)g_malloc((sizeof *leaf)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
node = &leaf[0];
|
|
|
|
fds->ptree_root = leaf;
|
|
|
|
} else if (fds->count < NODES_PER_LEVEL) {
|
|
|
|
/* It's a 1-level tree, and is going to stay that way for now. */
|
2013-03-02 09:06:03 +00:00
|
|
|
leaf = (frame_data *)fds->ptree_root;
|
2011-04-27 02:54:44 +00:00
|
|
|
node = &leaf[fds->count];
|
|
|
|
} else if (fds->count == NODES_PER_LEVEL) {
|
|
|
|
/* It's a 1-level tree that will turn into a 2-level tree. */
|
2013-03-02 09:06:03 +00:00
|
|
|
level1 = (frame_data **)g_malloc0((sizeof *level1)*NODES_PER_LEVEL);
|
|
|
|
level1[0] = (frame_data *)fds->ptree_root;
|
|
|
|
leaf = (frame_data *)g_malloc((sizeof *leaf)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level1[1] = leaf;
|
|
|
|
node = &leaf[0];
|
|
|
|
fds->ptree_root = level1;
|
|
|
|
} else if (fds->count < NODES_PER_LEVEL*NODES_PER_LEVEL) {
|
|
|
|
/* It's a 2-level tree, and is going to stay that way for now. */
|
2013-03-02 09:06:03 +00:00
|
|
|
level1 = (frame_data **)fds->ptree_root;
|
2011-04-27 02:54:44 +00:00
|
|
|
leaf = level1[fds->count >> LOG2_NODES_PER_LEVEL];
|
|
|
|
if (leaf == NULL) {
|
2013-03-02 09:06:03 +00:00
|
|
|
leaf = (frame_data *)g_malloc((sizeof *leaf)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level1[fds->count >> LOG2_NODES_PER_LEVEL] = leaf;
|
|
|
|
}
|
|
|
|
node = &leaf[LEAF_INDEX(fds->count)];
|
|
|
|
} else if (fds->count == NODES_PER_LEVEL*NODES_PER_LEVEL) {
|
|
|
|
/* It's a 2-level tree that will turn into a 3-level tree */
|
2013-03-02 09:06:03 +00:00
|
|
|
level2 = (frame_data ***)g_malloc0((sizeof *level2)*NODES_PER_LEVEL);
|
|
|
|
level2[0] = (frame_data **)fds->ptree_root;
|
|
|
|
level1 = (frame_data **)g_malloc0((sizeof *level1)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level2[1] = level1;
|
2013-03-02 09:06:03 +00:00
|
|
|
leaf = (frame_data *)g_malloc((sizeof *leaf)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level1[0] = leaf;
|
|
|
|
node = &leaf[0];
|
|
|
|
fds->ptree_root = level2;
|
|
|
|
} else if (fds->count < NODES_PER_LEVEL*NODES_PER_LEVEL*NODES_PER_LEVEL) {
|
|
|
|
/* It's a 3-level tree, and is going to stay that way for now. */
|
2013-03-02 09:06:03 +00:00
|
|
|
level2 = (frame_data ***)fds->ptree_root;
|
2011-04-27 02:54:44 +00:00
|
|
|
level1 = level2[fds->count >> (LOG2_NODES_PER_LEVEL+LOG2_NODES_PER_LEVEL)];
|
|
|
|
if (level1 == NULL) {
|
2013-03-02 09:06:03 +00:00
|
|
|
level1 = (frame_data **)g_malloc0((sizeof *level1)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level2[fds->count >> (LOG2_NODES_PER_LEVEL+LOG2_NODES_PER_LEVEL)] = level1;
|
|
|
|
}
|
|
|
|
leaf = level1[LEVEL_1_INDEX(fds->count)];
|
|
|
|
if (leaf == NULL) {
|
2013-03-02 09:06:03 +00:00
|
|
|
leaf = (frame_data *)g_malloc((sizeof *leaf)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level1[LEVEL_1_INDEX(fds->count)] = leaf;
|
|
|
|
}
|
|
|
|
node = &leaf[LEAF_INDEX(fds->count)];
|
|
|
|
} else if (fds->count == NODES_PER_LEVEL*NODES_PER_LEVEL*NODES_PER_LEVEL) {
|
|
|
|
/* It's a 3-level tree that will turn into a 4-level tree */
|
2013-03-02 09:06:03 +00:00
|
|
|
level3 = (frame_data ****)g_malloc0((sizeof *level3)*NODES_PER_LEVEL);
|
|
|
|
level3[0] = (frame_data ***)fds->ptree_root;
|
|
|
|
level2 = (frame_data ***)g_malloc0((sizeof *level2)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level3[1] = level2;
|
2013-03-02 09:06:03 +00:00
|
|
|
level1 = (frame_data **)g_malloc0((sizeof *level1)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level2[0] = level1;
|
2013-03-02 09:06:03 +00:00
|
|
|
leaf = (frame_data *)g_malloc((sizeof *leaf)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level1[0] = leaf;
|
|
|
|
node = &leaf[0];
|
|
|
|
fds->ptree_root = level3;
|
|
|
|
} else {
|
|
|
|
/* fds->count is 2^32-1 at most, and NODES_PER_LEVEL^4
|
|
|
|
2^(LOG2_NODES_PER_LEVEL*4), and LOG2_NODES_PER_LEVEL is 10,
|
|
|
|
so fds->count is always less < NODES_PER_LEVEL^4.
|
|
|
|
|
|
|
|
XXX - we should fail if fds->count is 2^31-1, or should
|
|
|
|
make the frame numbers 64-bit and just let users run
|
|
|
|
themselves out of address space or swap space. :-) */
|
|
|
|
/* It's a 4-level tree, and is going to stay that way forever. */
|
2013-03-02 09:06:03 +00:00
|
|
|
level3 = (frame_data ****)fds->ptree_root;
|
2011-04-27 02:54:44 +00:00
|
|
|
level2 = level3[LEVEL_3_INDEX(fds->count)];
|
|
|
|
if (level2 == NULL) {
|
2013-03-02 09:06:03 +00:00
|
|
|
level2 = (frame_data ***)g_malloc0((sizeof *level2)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level3[LEVEL_3_INDEX(fds->count)] = level2;
|
|
|
|
}
|
|
|
|
level1 = level2[LEVEL_2_INDEX(fds->count)];
|
|
|
|
if (level1 == NULL) {
|
2013-03-02 09:06:03 +00:00
|
|
|
level1 = (frame_data **)g_malloc0((sizeof *level1)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level2[LEVEL_2_INDEX(fds->count)] = level1;
|
|
|
|
}
|
|
|
|
leaf = level1[LEVEL_1_INDEX(fds->count)];
|
|
|
|
if (leaf == NULL) {
|
2013-03-02 09:06:03 +00:00
|
|
|
leaf = (frame_data *)g_malloc((sizeof *leaf)*NODES_PER_LEVEL);
|
2011-04-27 02:54:44 +00:00
|
|
|
level1[LEVEL_1_INDEX(fds->count)] = leaf;
|
|
|
|
}
|
|
|
|
node = &leaf[LEAF_INDEX(fds->count)];
|
|
|
|
}
|
|
|
|
*node = *fdata;
|
|
|
|
fds->count++;
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Find the frame_data for the specified frame number.
|
|
|
|
*/
|
|
|
|
frame_data *
|
|
|
|
frame_data_sequence_find(frame_data_sequence *fds, guint32 num)
|
|
|
|
{
|
|
|
|
frame_data *leaf;
|
|
|
|
frame_data **level1;
|
|
|
|
frame_data ***level2;
|
|
|
|
frame_data ****level3;
|
|
|
|
|
|
|
|
if (num == 0) {
|
|
|
|
/* There is no frame number 0 */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Convert it into an index number. */
|
|
|
|
num--;
|
|
|
|
if (num >= fds->count) {
|
|
|
|
/* There aren't that many frames. */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fds->count <= NODES_PER_LEVEL) {
|
|
|
|
/* It's a 1-level tree. */
|
2013-03-02 09:06:03 +00:00
|
|
|
leaf = (frame_data *)fds->ptree_root;
|
2011-04-27 02:54:44 +00:00
|
|
|
return &leaf[num];
|
|
|
|
}
|
|
|
|
if (fds->count <= NODES_PER_LEVEL*NODES_PER_LEVEL) {
|
|
|
|
/* It's a 2-level tree. */
|
2013-03-02 09:06:03 +00:00
|
|
|
level1 = (frame_data **)fds->ptree_root;
|
2011-04-27 02:54:44 +00:00
|
|
|
leaf = level1[num >> LOG2_NODES_PER_LEVEL];
|
|
|
|
return &leaf[LEAF_INDEX(num)];
|
|
|
|
}
|
|
|
|
if (fds->count <= NODES_PER_LEVEL*NODES_PER_LEVEL*NODES_PER_LEVEL) {
|
|
|
|
/* It's a 3-level tree. */
|
2013-03-02 09:06:03 +00:00
|
|
|
level2 = (frame_data ***)fds->ptree_root;
|
2011-04-27 02:54:44 +00:00
|
|
|
level1 = level2[num >> (LOG2_NODES_PER_LEVEL+LOG2_NODES_PER_LEVEL)];
|
|
|
|
leaf = level1[(num >> LOG2_NODES_PER_LEVEL) & (NODES_PER_LEVEL - 1)];
|
|
|
|
return &leaf[LEAF_INDEX(num)];
|
|
|
|
}
|
|
|
|
/* fds->count is 2^32-1 at most, and NODES_PER_LEVEL^4
|
|
|
|
2^(LOG2_NODES_PER_LEVEL*4), and LOG2_NODES_PER_LEVEL is 10,
|
|
|
|
so fds->count is always less < NODES_PER_LEVEL^4. */
|
|
|
|
/* It's a 4-level tree, and is going to stay that way forever. */
|
2013-03-02 09:06:03 +00:00
|
|
|
level3 = (frame_data ****)fds->ptree_root;
|
2011-04-27 02:54:44 +00:00
|
|
|
level2 = level3[num >> (LOG2_NODES_PER_LEVEL+LOG2_NODES_PER_LEVEL+LOG2_NODES_PER_LEVEL)];
|
|
|
|
level1 = level2[(num >> (LOG2_NODES_PER_LEVEL+LOG2_NODES_PER_LEVEL)) & (NODES_PER_LEVEL - 1)];
|
|
|
|
leaf = level1[(num >> LOG2_NODES_PER_LEVEL) & (NODES_PER_LEVEL - 1)];
|
|
|
|
return &leaf[LEAF_INDEX(num)];
|
|
|
|
}
|
|
|
|
|
2013-03-03 03:26:09 +00:00
|
|
|
/* recursively frees a frame_data radix level */
|
2013-05-22 20:14:16 +00:00
|
|
|
static void
|
2013-03-03 03:26:09 +00:00
|
|
|
free_frame_data_array(void *array, guint count, guint level, gboolean last)
|
|
|
|
{
|
|
|
|
guint i, level_count;
|
|
|
|
|
|
|
|
if (last) {
|
|
|
|
/* if we are the last in our given parent's row, we may not have
|
|
|
|
* exactly a full row, so do the bit twiddling to figure out exactly
|
|
|
|
* how many fields we have */
|
|
|
|
level_count = (count >> ((level - 1) * LOG2_NODES_PER_LEVEL)) &
|
|
|
|
(NODES_PER_LEVEL - 1);
|
|
|
|
/* the above calculation rounds down, so make sure we count correctly
|
|
|
|
* if count is not an even multiple of NODES_PER_LEVEL */
|
|
|
|
if (count & ((1 << ((level - 1) * LOG2_NODES_PER_LEVEL)) - 1)) {
|
|
|
|
level_count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* if we're not the last in our parent, then we're guaranteed to have
|
|
|
|
* a full array */
|
|
|
|
level_count = NODES_PER_LEVEL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (level > 1) {
|
|
|
|
/* recurse on every sub-array, passing on our own 'last' value
|
|
|
|
* specially to our last child */
|
|
|
|
frame_data **real_array = (frame_data **) array;
|
|
|
|
|
|
|
|
for (i=0; i < level_count-1; i++) {
|
|
|
|
free_frame_data_array(real_array[i], count, level-1, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
free_frame_data_array(real_array[level_count-1], count, level-1, last);
|
|
|
|
}
|
|
|
|
else if (level == 1) {
|
|
|
|
/* bottom level, so just clean up all the frame data */
|
|
|
|
frame_data *real_array = (frame_data *) array;
|
|
|
|
|
|
|
|
for (i=0; i < level_count; i++) {
|
2013-03-15 18:04:50 +00:00
|
|
|
frame_data_destroy(&real_array[i]);
|
2013-03-03 03:26:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* free the array itself */
|
|
|
|
g_free(array);
|
|
|
|
}
|
|
|
|
|
2011-04-27 02:54:44 +00:00
|
|
|
/*
|
|
|
|
* Free a frame_data_sequence and all the frame_data structures in it.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
free_frame_data_sequence(frame_data_sequence *fds)
|
|
|
|
{
|
2013-03-03 03:26:09 +00:00
|
|
|
guint32 count = fds->count;
|
|
|
|
guint levels = 0;
|
2011-04-27 02:54:44 +00:00
|
|
|
|
2013-03-03 03:26:09 +00:00
|
|
|
/* calculate how many levels we have */
|
|
|
|
while (count) {
|
|
|
|
levels++;
|
|
|
|
count >>= LOG2_NODES_PER_LEVEL;
|
2011-04-27 02:54:44 +00:00
|
|
|
}
|
2013-03-03 03:26:09 +00:00
|
|
|
|
|
|
|
/* call the recursive free function */
|
|
|
|
if (levels > 0) {
|
|
|
|
free_frame_data_array(fds->ptree_root, fds->count, levels, TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* free the header struct */
|
2011-04-27 02:54:44 +00:00
|
|
|
g_free(fds);
|
|
|
|
}
|
2013-03-03 03:26:09 +00:00
|
|
|
|
2013-03-27 17:14:42 +00:00
|
|
|
void
|
|
|
|
find_and_mark_frame_depended_upon(gpointer data, gpointer user_data)
|
|
|
|
{
|
|
|
|
frame_data *dependent_fd;
|
|
|
|
guint32 dependent_frame = GPOINTER_TO_UINT(data);
|
|
|
|
frame_data_sequence *frames = (frame_data_sequence *)user_data;
|
|
|
|
|
|
|
|
if (dependent_frame && frames) {
|
|
|
|
dependent_fd = frame_data_sequence_find(frames, dependent_frame);
|
|
|
|
dependent_fd->flags.dependent_of_displayed = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-03 03:26:09 +00:00
|
|
|
/*
|
|
|
|
* Editor modelines - http://www.wireshark.org/tools/modelines.html
|
|
|
|
*
|
|
|
|
* Local variables:
|
|
|
|
* c-basic-offset: 2
|
|
|
|
* tab-width: 8
|
|
|
|
* indent-tabs-mode: nil
|
|
|
|
* End:
|
|
|
|
*
|
|
|
|
* vi: set shiftwidth=2 tabstop=8 expandtab:
|
|
|
|
* :indentSize=2:tabSize=8:noTabs=true:
|
|
|
|
*/
|