aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGuy Harris <guy@alum.mit.edu>2012-07-07 08:15:41 +0000
committerGuy Harris <guy@alum.mit.edu>2012-07-07 08:15:41 +0000
commit85f219be2e41191971218a632022f590003f992c (patch)
tree83e8e6d8fc2e605d78ee7497db3f4e2587fbd013
parentef825ec6d3e07e97a85d52e116338c08c2637432 (diff)
From Michael Mann:
Add a preference for the packet length statistics. Fixes bug 3239. svn path=/trunk/; revision=43597
-rw-r--r--doc/README.stats_tree1
-rw-r--r--epan/libwireshark.def2
-rw-r--r--epan/prefs.c32
-rw-r--r--epan/prefs.h14
-rw-r--r--epan/stats_tree.c17
-rw-r--r--epan/stats_tree.h6
-rw-r--r--plugins/stats_tree/pinfo_stats_tree.c101
7 files changed, 170 insertions, 3 deletions
diff --git a/doc/README.stats_tree b/doc/README.stats_tree
index d0a30c3cbc..fa6fc4fdd4 100644
--- a/doc/README.stats_tree
+++ b/doc/README.stats_tree
@@ -158,6 +158,7 @@ stats_tree_create_node_by_pname(st, name, parent_name, with_children);
stats_tree_create_range_node(st, name, parent_id, ...)
+stats_tree_create_range_node_string(st, name, parent_id, num_str_ranges, str_ranges)
stats_tree_range_node_with_pname(st, name, parent_name, ...)
Creates a node in the tree, that will contain a ranges list.
example:
diff --git a/epan/libwireshark.def b/epan/libwireshark.def
index aa0fb2e6cb..52b2eff149 100644
--- a/epan/libwireshark.def
+++ b/epan/libwireshark.def
@@ -738,6 +738,7 @@ prefs_register_modules
prefs_register_obsolete_preference
prefs_register_protocol
prefs_register_protocol_subtree
+prefs_register_stat
prefs_register_range_preference
prefs_register_static_text_preference
prefs_register_string_preference
@@ -1008,6 +1009,7 @@ stats_tree_create_node_by_pname
stats_tree_create_pivot_by_pname
stats_tree_create_pivot
stats_tree_create_range_node
+stats_tree_create_range_node_string
stats_tree_free
stats_tree_get_abbr
stats_tree_get_cfg_by_abbr
diff --git a/epan/prefs.c b/epan/prefs.c
index bfbcad9c14..1ac65cdb7d 100644
--- a/epan/prefs.c
+++ b/epan/prefs.c
@@ -319,7 +319,6 @@ prefs_register_module_or_subtree(module_t *parent, const char *name,
* Register that a protocol has preferences.
*/
module_t *protocols_module = NULL;
-module_t *stats_module = NULL;
module_t *
prefs_register_protocol(int id, void (*apply_cb)(void))
@@ -427,6 +426,37 @@ prefs_register_protocol_obsolete(int id)
return module;
}
+/*
+ * Register that a statistical tap has preferences.
+ *
+ * "name" is a name for the tap to use on the command line with "-o"
+ * and in preference files.
+ *
+ * "title" is a short human-readable name for the tap.
+ *
+ * "description" is a longer human-readable description of the tap.
+ */
+module_t *stats_module = NULL;
+
+module_t *
+prefs_register_stat(const char *name, const char *title,
+ const char *description, void (*apply_cb)(void))
+{
+ /*
+ * Have we yet created the "Statistics" subtree?
+ */
+ if (stats_module == NULL) {
+ /*
+ * No. Register Statistics subtree as well as any preferences
+ * for non-dissector modules.
+ */
+ prefs_register_modules();
+ }
+
+ return prefs_register_module(stats_module, name, title, description,
+ apply_cb);
+}
+
module_t *
prefs_find_module(const char *name)
{
diff --git a/epan/prefs.h b/epan/prefs.h
index 066f136a73..b7ea7d8025 100644
--- a/epan/prefs.h
+++ b/epan/prefs.h
@@ -221,6 +221,20 @@ extern module_t *prefs_register_subtree(module_t *parent, const char *title,
extern module_t *prefs_register_protocol(int id, void (*apply_cb)(void));
/*
+ * Register that a statistical tap has preferences.
+ *
+ * "name" is a name for the tap to use on the command line with "-o"
+ * and in preference files.
+ *
+ * "title" is a short human-readable name for the tap.
+ *
+ * "description" is a longer human-readable description of the tap.
+ */
+extern module_t *prefs_register_stat(const char *name, const char *title,
+ const char *description,
+ void (*apply_cb)(void));
+
+/*
* Register that a protocol has preferences and group it under a single
* subtree
*/
diff --git a/epan/stats_tree.c b/epan/stats_tree.c
index f0e4d690ac..6e1c52d56e 100644
--- a/epan/stats_tree.c
+++ b/epan/stats_tree.c
@@ -648,6 +648,23 @@ stats_tree_create_range_node(stats_tree *st, const gchar *name, int parent_id, .
return rng_root->id;
}
+extern int
+stats_tree_create_range_node_string(stats_tree *st, const gchar *name,
+ int parent_id, int num_str_ranges,
+ gchar** str_ranges)
+{
+ int i;
+ stat_node *rng_root = new_stat_node(st, name, parent_id, FALSE, TRUE);
+ stat_node *range_node = NULL;
+
+ for (i = 0; i < num_str_ranges; i++) {
+ range_node = new_stat_node(st, str_ranges[i], rng_root->id, FALSE, FALSE);
+ range_node->rng = get_range(str_ranges[i]);
+ }
+
+ return rng_root->id;
+}
+
/****/
extern int
stats_tree_parent_id_by_name(stats_tree *st, const gchar *parent_name)
diff --git a/epan/stats_tree.h b/epan/stats_tree.h
index ebc9953fab..98c6309d20 100644
--- a/epan/stats_tree.h
+++ b/epan/stats_tree.h
@@ -113,6 +113,12 @@ extern int stats_tree_create_range_node(stats_tree *st,
int parent_id,
...);
+extern int stats_tree_create_range_node_string(stats_tree *st,
+ const gchar *name,
+ int parent_id,
+ int num_str_ranges,
+ gchar** str_ranges);
+
extern int stats_tree_range_node_with_pname(stats_tree *st,
const gchar *name,
const gchar *parent_name,
diff --git a/plugins/stats_tree/pinfo_stats_tree.c b/plugins/stats_tree/pinfo_stats_tree.c
index 5693b04333..4d9aa351e3 100644
--- a/plugins/stats_tree/pinfo_stats_tree.c
+++ b/plugins/stats_tree/pinfo_stats_tree.c
@@ -29,6 +29,9 @@
#endif
#include <epan/stats_tree.h>
+#include <epan/prefs.h>
+#include <epan/uat.h>
+#include <epan/uat-int.h>
#include "pinfo_stats_tree.h"
@@ -45,10 +48,70 @@ static const gchar* port_type_to_str (port_type type) {
case PT_DDP: return "DDP";
case PT_SBCCS: return "FICON SBCCS";
case PT_IDP: return "IDP";
- default: return "[Unknown]";
+ default: return "[Unknown]";
}
}
+/*-------------------------------------
+ * UAT for Packet Lengths
+ *-------------------------------------
+ */
+typedef struct {
+ range_t *packet_range;
+} uat_plen_record_t;
+
+static range_t default_range[10] = {
+ {1, {{0, 19}}},
+ {1, {{20, 39}}},
+ {1, {{40, 79}}},
+ {1, {{80, 159}}},
+ {1, {{160, 319}}},
+ {1, {{320, 639}}},
+ {1, {{640, 1279}}},
+ {1, {{1280, 2559}}},
+ {1, {{2560, 5119}}},
+ {1, {{5120, 0xFFFFFFFF}}}
+};
+static uat_plen_record_t *uat_plen_records = NULL;
+static uat_t * plen_uat = NULL;
+static guint num_plen_uat = 0;
+
+static void* uat_plen_record_copy_cb(void* n, const void* o, size_t siz _U_) {
+ const uat_plen_record_t *r = o;
+ uat_plen_record_t *rn = n;
+
+ if (r->packet_range)
+ rn->packet_range = range_copy(r->packet_range);
+
+ return n;
+}
+
+static void uat_plen_record_free_cb(void*r) {
+ uat_plen_record_t* record = (uat_plen_record_t*)r;
+
+ if (record->packet_range)
+ g_free(record->packet_range);
+}
+
+static void uat_plen_record_post_update_cb(void) {
+ guint i, num_default;
+ uat_plen_record_t rec;
+
+ /* If there are no records, create default list */
+ if (num_plen_uat == 0) {
+ num_default = sizeof(default_range)/sizeof(range_t);
+
+ /* default values for packet lengths */
+ for (i = 0; i < num_default; i++)
+ {
+ rec.packet_range = &default_range[i];
+ uat_add_record(plen_uat, &rec);
+ }
+ }
+}
+
+UAT_RANGE_CB_DEF(uat_plen_records, packet_range, uat_plen_record_t)
+
/* ip host stats_tree -- basic test */
static int st_node_ip = -1;
static const gchar* st_str_ip = "IP Addresses";
@@ -88,7 +151,15 @@ static int st_node_plen = -1;
static const gchar* st_str_plen = "Packet Lengths";
static void plen_stats_tree_init(stats_tree* st) {
- st_node_plen = stats_tree_create_range_node(st, st_str_plen, 0, "0-19","20-39","40-79","80-159","160-319","320-639","640-1279","1280-2559","2560-5119","5120-",NULL);
+ guint i;
+ char **str_range_array = ep_alloc(num_plen_uat*sizeof(char*));
+
+ /* Convert the ranges to strings for the stats tree API */
+ for (i = 0; i < num_plen_uat; i++) {
+ str_range_array[i] = range_convert_range(uat_plen_records[i].packet_range);
+ }
+
+ st_node_plen = stats_tree_create_range_node_string(st, st_str_plen, 0, num_plen_uat, str_range_array);
}
static int plen_stats_tree_packet(stats_tree* st, packet_info* pinfo, epan_dissect_t *edt _U_, const void *p _U_) {
@@ -130,9 +201,35 @@ static int dsts_stats_tree_packet(stats_tree* st, packet_info* pinfo, epan_disse
/* register all pinfo trees */
void register_pinfo_stat_trees(void) {
+ module_t *stat_module;
+
+ static uat_field_t plen_uat_flds[] = {
+ UAT_FLD_RANGE(uat_plen_records, packet_range, "Packet Range", 0xFFFFFFFF, "Range of packet sizes to count"),
+ UAT_END_FIELDS
+ };
+
stats_tree_register("ip","ip_hosts",st_str_ip, 0, ip_hosts_stats_tree_packet, ip_hosts_stats_tree_init, NULL );
stats_tree_register("ip","ptype",st_str_ptype, 0, ptype_stats_tree_packet, ptype_stats_tree_init, NULL );
stats_tree_register_with_group("frame","plen",st_str_plen, 0, plen_stats_tree_packet, plen_stats_tree_init, NULL, REGISTER_STAT_GROUP_GENERIC );
stats_tree_register("ip","dests",st_str_dsts, 0, dsts_stats_tree_packet, dsts_stats_tree_init, NULL );
+
+ stat_module = prefs_register_stat("stat_tree", "Stats Tree", "Stats Tree", NULL);
+
+ plen_uat = uat_new("Packet Lengths",
+ sizeof(uat_plen_record_t), /* record size */
+ "packet_lengths", /* filename */
+ TRUE, /* from_profile */
+ (void*) &uat_plen_records, /* data_ptr */
+ &num_plen_uat, /* numitems_ptr */
+ UAT_CAT_GENERAL, /* category */
+ NULL, /* help */
+ uat_plen_record_copy_cb, /* copy callback */
+ NULL, /* update callback */
+ uat_plen_record_free_cb, /* free callback */
+ uat_plen_record_post_update_cb, /* post update callback */
+ plen_uat_flds); /* UAT field definitions */
+
+ prefs_register_uat_preference(stat_module, "packet_lengths",
+ "Packet Lengths", "Delineated packet sizes to count", plen_uat);
}