1998-09-16 02:39:15 +00:00
|
|
|
/* print.c
|
|
|
|
* Routines for printing packet analysis trees.
|
|
|
|
*
|
2002-08-28 21:04:11 +00:00
|
|
|
* $Id: print.c,v 1.57 2002/08/28 21:00:40 jmayer Exp $
|
1998-09-16 03:22:19 +00:00
|
|
|
*
|
2001-11-13 23:55:44 +00:00
|
|
|
* Gilbert Ramirez <gram@alumni.rice.edu>
|
1998-09-16 02:39:15 +00:00
|
|
|
*
|
|
|
|
* Ethereal - Network traffic analyzer
|
2001-06-08 08:50:51 +00:00
|
|
|
* By Gerald Combs <gerald@ethereal.com>
|
1998-09-16 02:39:15 +00:00
|
|
|
* Copyright 1998 Gerald Combs
|
2002-08-28 21:04:11 +00:00
|
|
|
*
|
1998-09-16 02:39:15 +00:00
|
|
|
* 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.
|
2002-08-28 21:04:11 +00:00
|
|
|
*
|
1998-09-16 02:39:15 +00:00
|
|
|
* 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.
|
2002-08-28 21:04:11 +00:00
|
|
|
*
|
1998-09-16 02:39:15 +00:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2002-06-04 07:03:57 +00:00
|
|
|
#include <epan/epan.h>
|
|
|
|
#include <epan/epan_dissect.h>
|
|
|
|
#include <epan/tvbuff.h>
|
2002-01-21 07:37:49 +00:00
|
|
|
#include <epan/packet.h>
|
2002-06-04 07:03:57 +00:00
|
|
|
|
1998-09-16 02:39:15 +00:00
|
|
|
#include "print.h"
|
1998-09-27 22:12:47 +00:00
|
|
|
#include "ps.h"
|
1999-11-22 08:03:31 +00:00
|
|
|
#include "util.h"
|
2002-05-10 23:20:38 +00:00
|
|
|
#include "packet-data.h"
|
1998-09-16 02:39:15 +00:00
|
|
|
|
2002-06-22 01:43:57 +00:00
|
|
|
static void proto_tree_print_node(GNode *node, gpointer data);
|
2002-08-02 23:36:07 +00:00
|
|
|
static void print_hex_data_buffer(FILE *fh, register const guchar *cp,
|
|
|
|
register guint length, char_enc encoding, gint format);
|
1998-09-16 02:39:15 +00:00
|
|
|
static void ps_clean_string(unsigned char *out, const unsigned char *in,
|
|
|
|
int outbuf_size);
|
|
|
|
|
When printing a packet, do it from the protocol tree, not from the GTK+
tree constructed from the protocol tree:
1) The value of "level" field of GTK+ tree items appears to
depend on various random things - see a change I made to
"packet-dns.c" a while ago, to change the order in which
items were put in the tree, so that DNS trees printed with
correct indentation - and, right now, we appear to be doing
*something* wrong, as some packets I printed from one file
here had randomly bogus indentation; I could probably track
the problem down and fix it, but that might just hold us
until we accidentally do something *else* wrong by GTK+'s
lights.
The new code provides its own tree level as it goes.
2) The new code is independent of GTK+, so it could be used with
other toolkits, or with non-GUI variants of Ethereal.
3) This may make it easier to add a "Print..." menu item to let
the user print packets other than the currently selected
packet.
Make the internal routines used to print the packet static.
For the "Print Packet" menu item, put up a message box if they haven't
yet selected a packet.
svn path=/trunk/; revision=362
1999-07-13 04:38:15 +00:00
|
|
|
typedef struct {
|
|
|
|
int level;
|
|
|
|
FILE *fh;
|
2001-06-08 08:50:51 +00:00
|
|
|
GSList *src_list;
|
1999-09-12 20:23:43 +00:00
|
|
|
gboolean print_all_levels;
|
1999-09-29 22:19:24 +00:00
|
|
|
gboolean print_hex_for_data;
|
1999-11-22 06:24:56 +00:00
|
|
|
char_enc encoding;
|
2002-06-22 01:43:57 +00:00
|
|
|
gint format; /* text or PostScript */
|
When printing a packet, do it from the protocol tree, not from the GTK+
tree constructed from the protocol tree:
1) The value of "level" field of GTK+ tree items appears to
depend on various random things - see a change I made to
"packet-dns.c" a while ago, to change the order in which
items were put in the tree, so that DNS trees printed with
correct indentation - and, right now, we appear to be doing
*something* wrong, as some packets I printed from one file
here had randomly bogus indentation; I could probably track
the problem down and fix it, but that might just hold us
until we accidentally do something *else* wrong by GTK+'s
lights.
The new code provides its own tree level as it goes.
2) The new code is independent of GTK+, so it could be used with
other toolkits, or with non-GUI variants of Ethereal.
3) This may make it easier to add a "Print..." menu item to let
the user print packets other than the currently selected
packet.
Make the internal routines used to print the packet static.
For the "Print Packet" menu item, put up a message box if they haven't
yet selected a packet.
svn path=/trunk/; revision=362
1999-07-13 04:38:15 +00:00
|
|
|
} print_data;
|
|
|
|
|
1999-07-23 08:29:24 +00:00
|
|
|
FILE *open_print_dest(int to_file, const char *dest)
|
1998-09-16 02:39:15 +00:00
|
|
|
{
|
|
|
|
FILE *fh;
|
|
|
|
|
|
|
|
/* Open the file or command for output */
|
1999-07-23 08:29:24 +00:00
|
|
|
if (to_file)
|
|
|
|
fh = fopen(dest, "w");
|
|
|
|
else
|
1999-07-23 08:30:57 +00:00
|
|
|
fh = popen(dest, "w");
|
1998-09-16 02:39:15 +00:00
|
|
|
|
1999-07-23 08:29:24 +00:00
|
|
|
return fh;
|
|
|
|
}
|
|
|
|
|
|
|
|
void close_print_dest(int to_file, FILE *fh)
|
|
|
|
{
|
|
|
|
/* Close the file or command */
|
|
|
|
if (to_file)
|
|
|
|
fclose(fh);
|
|
|
|
else
|
|
|
|
pclose(fh);
|
|
|
|
}
|
|
|
|
|
2002-06-04 07:03:57 +00:00
|
|
|
void proto_tree_print(print_args_t *print_args, epan_dissect_t *edt,
|
|
|
|
FILE *fh)
|
1999-07-23 08:29:24 +00:00
|
|
|
{
|
|
|
|
print_data data;
|
1998-09-16 02:39:15 +00:00
|
|
|
|
|
|
|
/* Create the output */
|
When printing a packet, do it from the protocol tree, not from the GTK+
tree constructed from the protocol tree:
1) The value of "level" field of GTK+ tree items appears to
depend on various random things - see a change I made to
"packet-dns.c" a while ago, to change the order in which
items were put in the tree, so that DNS trees printed with
correct indentation - and, right now, we appear to be doing
*something* wrong, as some packets I printed from one file
here had randomly bogus indentation; I could probably track
the problem down and fix it, but that might just hold us
until we accidentally do something *else* wrong by GTK+'s
lights.
The new code provides its own tree level as it goes.
2) The new code is independent of GTK+, so it could be used with
other toolkits, or with non-GUI variants of Ethereal.
3) This may make it easier to add a "Print..." menu item to let
the user print packets other than the currently selected
packet.
Make the internal routines used to print the packet static.
For the "Print Packet" menu item, put up a message box if they haven't
yet selected a packet.
svn path=/trunk/; revision=362
1999-07-13 04:38:15 +00:00
|
|
|
data.level = 0;
|
|
|
|
data.fh = fh;
|
2002-06-04 07:03:57 +00:00
|
|
|
data.src_list = edt->pi.data_src;
|
|
|
|
data.encoding = edt->pi.fd->flags.encoding;
|
1999-09-29 22:19:24 +00:00
|
|
|
data.print_all_levels = print_args->expand_all;
|
|
|
|
data.print_hex_for_data = !print_args->print_hex;
|
|
|
|
/* If we're printing the entire packet in hex, don't
|
|
|
|
print uninterpreted data fields in hex as well. */
|
2002-06-22 01:43:57 +00:00
|
|
|
data.format = print_args->format;
|
1999-07-23 21:09:25 +00:00
|
|
|
|
2002-06-22 01:43:57 +00:00
|
|
|
g_node_children_foreach((GNode*) edt->tree, G_TRAVERSE_ALL,
|
|
|
|
proto_tree_print_node, &data);
|
1998-09-16 02:39:15 +00:00
|
|
|
}
|
|
|
|
|
2001-06-08 08:50:51 +00:00
|
|
|
/*
|
2002-02-18 01:08:44 +00:00
|
|
|
* Find the data source for a specified field, and return a pointer
|
|
|
|
* to the data in it.
|
2001-06-08 08:50:51 +00:00
|
|
|
*/
|
|
|
|
static const guint8 *
|
2001-06-08 10:07:55 +00:00
|
|
|
get_field_data(GSList *src_list, field_info *fi)
|
2001-06-08 08:50:51 +00:00
|
|
|
{
|
2002-02-18 01:08:44 +00:00
|
|
|
GSList *src_le;
|
|
|
|
data_source *src;
|
2001-06-08 08:50:51 +00:00
|
|
|
tvbuff_t *src_tvb;
|
|
|
|
|
2002-02-18 01:08:44 +00:00
|
|
|
for (src_le = src_list; src_le != NULL; src_le = src_le->next) {
|
|
|
|
src = src_le->data;
|
|
|
|
src_tvb = src->tvb;
|
|
|
|
if (fi->ds_tvb == src_tvb) {
|
2001-06-08 08:50:51 +00:00
|
|
|
/*
|
|
|
|
* Found it.
|
|
|
|
*/
|
2001-06-08 10:07:55 +00:00
|
|
|
return tvb_get_ptr(src_tvb, fi->start, fi->length);
|
2001-06-08 08:50:51 +00:00
|
|
|
}
|
|
|
|
}
|
2001-06-08 10:07:55 +00:00
|
|
|
g_assert_not_reached();
|
2001-06-08 08:50:51 +00:00
|
|
|
return NULL; /* not found */
|
|
|
|
}
|
|
|
|
|
2002-03-14 05:41:59 +00:00
|
|
|
#define MAX_INDENT 160
|
|
|
|
|
2002-06-22 01:43:57 +00:00
|
|
|
#define MAX_PS_LINE_LENGTH 256
|
|
|
|
|
|
|
|
/* Print a tree's data, and any child nodes. */
|
When printing a packet, do it from the protocol tree, not from the GTK+
tree constructed from the protocol tree:
1) The value of "level" field of GTK+ tree items appears to
depend on various random things - see a change I made to
"packet-dns.c" a while ago, to change the order in which
items were put in the tree, so that DNS trees printed with
correct indentation - and, right now, we appear to be doing
*something* wrong, as some packets I printed from one file
here had randomly bogus indentation; I could probably track
the problem down and fix it, but that might just hold us
until we accidentally do something *else* wrong by GTK+'s
lights.
The new code provides its own tree level as it goes.
2) The new code is independent of GTK+, so it could be used with
other toolkits, or with non-GUI variants of Ethereal.
3) This may make it easier to add a "Print..." menu item to let
the user print packets other than the currently selected
packet.
Make the internal routines used to print the packet static.
For the "Print Packet" menu item, put up a message box if they haven't
yet selected a packet.
svn path=/trunk/; revision=362
1999-07-13 04:38:15 +00:00
|
|
|
static
|
2002-06-22 01:43:57 +00:00
|
|
|
void proto_tree_print_node(GNode *node, gpointer data)
|
1998-09-16 02:39:15 +00:00
|
|
|
{
|
2001-12-18 21:31:02 +00:00
|
|
|
field_info *fi = PITEM_FINFO(node);
|
When printing a packet, do it from the protocol tree, not from the GTK+
tree constructed from the protocol tree:
1) The value of "level" field of GTK+ tree items appears to
depend on various random things - see a change I made to
"packet-dns.c" a while ago, to change the order in which
items were put in the tree, so that DNS trees printed with
correct indentation - and, right now, we appear to be doing
*something* wrong, as some packets I printed from one file
here had randomly bogus indentation; I could probably track
the problem down and fix it, but that might just hold us
until we accidentally do something *else* wrong by GTK+'s
lights.
The new code provides its own tree level as it goes.
2) The new code is independent of GTK+, so it could be used with
other toolkits, or with non-GUI variants of Ethereal.
3) This may make it easier to add a "Print..." menu item to let
the user print packets other than the currently selected
packet.
Make the internal routines used to print the packet static.
For the "Print Packet" menu item, put up a message box if they haven't
yet selected a packet.
svn path=/trunk/; revision=362
1999-07-13 04:38:15 +00:00
|
|
|
print_data *pdata = (print_data*) data;
|
2001-06-08 08:50:51 +00:00
|
|
|
const guint8 *pd;
|
When printing a packet, do it from the protocol tree, not from the GTK+
tree constructed from the protocol tree:
1) The value of "level" field of GTK+ tree items appears to
depend on various random things - see a change I made to
"packet-dns.c" a while ago, to change the order in which
items were put in the tree, so that DNS trees printed with
correct indentation - and, right now, we appear to be doing
*something* wrong, as some packets I printed from one file
here had randomly bogus indentation; I could probably track
the problem down and fix it, but that might just hold us
until we accidentally do something *else* wrong by GTK+'s
lights.
The new code provides its own tree level as it goes.
2) The new code is independent of GTK+, so it could be used with
other toolkits, or with non-GUI variants of Ethereal.
3) This may make it easier to add a "Print..." menu item to let
the user print packets other than the currently selected
packet.
Make the internal routines used to print the packet static.
For the "Print Packet" menu item, put up a message box if they haven't
yet selected a packet.
svn path=/trunk/; revision=362
1999-07-13 04:38:15 +00:00
|
|
|
gchar label_str[ITEM_LABEL_LENGTH];
|
|
|
|
gchar *label_ptr;
|
|
|
|
|
1999-09-12 20:23:43 +00:00
|
|
|
/* Don't print invisible entries. */
|
When printing a packet, do it from the protocol tree, not from the GTK+
tree constructed from the protocol tree:
1) The value of "level" field of GTK+ tree items appears to
depend on various random things - see a change I made to
"packet-dns.c" a while ago, to change the order in which
items were put in the tree, so that DNS trees printed with
correct indentation - and, right now, we appear to be doing
*something* wrong, as some packets I printed from one file
here had randomly bogus indentation; I could probably track
the problem down and fix it, but that might just hold us
until we accidentally do something *else* wrong by GTK+'s
lights.
The new code provides its own tree level as it goes.
2) The new code is independent of GTK+, so it could be used with
other toolkits, or with non-GUI variants of Ethereal.
3) This may make it easier to add a "Print..." menu item to let
the user print packets other than the currently selected
packet.
Make the internal routines used to print the packet static.
For the "Print Packet" menu item, put up a message box if they haven't
yet selected a packet.
svn path=/trunk/; revision=362
1999-07-13 04:38:15 +00:00
|
|
|
if (!fi->visible)
|
|
|
|
return;
|
1998-09-16 02:39:15 +00:00
|
|
|
|
When printing a packet, do it from the protocol tree, not from the GTK+
tree constructed from the protocol tree:
1) The value of "level" field of GTK+ tree items appears to
depend on various random things - see a change I made to
"packet-dns.c" a while ago, to change the order in which
items were put in the tree, so that DNS trees printed with
correct indentation - and, right now, we appear to be doing
*something* wrong, as some packets I printed from one file
here had randomly bogus indentation; I could probably track
the problem down and fix it, but that might just hold us
until we accidentally do something *else* wrong by GTK+'s
lights.
The new code provides its own tree level as it goes.
2) The new code is independent of GTK+, so it could be used with
other toolkits, or with non-GUI variants of Ethereal.
3) This may make it easier to add a "Print..." menu item to let
the user print packets other than the currently selected
packet.
Make the internal routines used to print the packet static.
For the "Print Packet" menu item, put up a message box if they haven't
yet selected a packet.
svn path=/trunk/; revision=362
1999-07-13 04:38:15 +00:00
|
|
|
/* was a free format label produced? */
|
|
|
|
if (fi->representation) {
|
|
|
|
label_ptr = fi->representation;
|
|
|
|
}
|
|
|
|
else { /* no, make a generic label */
|
|
|
|
label_ptr = label_str;
|
|
|
|
proto_item_fill_label(fi, label_str);
|
|
|
|
}
|
2002-08-28 21:04:11 +00:00
|
|
|
|
2002-06-29 09:45:06 +00:00
|
|
|
print_line(pdata->fh, pdata->level, pdata->format, label_ptr);
|
1998-09-16 02:39:15 +00:00
|
|
|
|
1999-09-29 22:19:24 +00:00
|
|
|
/* If it's uninterpreted data, dump it (unless our caller will
|
|
|
|
be printing the entire packet in hex). */
|
2001-06-08 08:50:51 +00:00
|
|
|
if (fi->hfinfo->id == proto_data && pdata->print_hex_for_data) {
|
|
|
|
/*
|
2001-06-08 10:07:55 +00:00
|
|
|
* Find the data for this field.
|
2001-06-08 08:50:51 +00:00
|
|
|
*/
|
2001-06-08 10:07:55 +00:00
|
|
|
pd = get_field_data(pdata->src_list, fi);
|
2002-06-22 01:24:23 +00:00
|
|
|
print_hex_data_buffer(pdata->fh, pd, fi->length,
|
2002-06-22 01:43:57 +00:00
|
|
|
pdata->encoding, pdata->format);
|
2001-06-08 08:50:51 +00:00
|
|
|
}
|
1998-09-16 02:39:15 +00:00
|
|
|
|
2002-04-02 05:07:36 +00:00
|
|
|
/* If we're printing all levels, or if this node is one with a
|
|
|
|
subtree and its subtree is expanded, recurse into the subtree,
|
|
|
|
if it exists. */
|
|
|
|
g_assert(fi->tree_type >= -1 && fi->tree_type < num_tree_types);
|
|
|
|
if (pdata->print_all_levels ||
|
|
|
|
(fi->tree_type >= 0 && tree_is_expanded[fi->tree_type])) {
|
1999-09-12 20:23:43 +00:00
|
|
|
if (g_node_n_children(node) > 0) {
|
|
|
|
pdata->level++;
|
|
|
|
g_node_children_foreach(node, G_TRAVERSE_ALL,
|
2002-06-22 01:43:57 +00:00
|
|
|
proto_tree_print_node, pdata);
|
1999-09-12 20:23:43 +00:00
|
|
|
pdata->level--;
|
|
|
|
}
|
1998-09-16 02:39:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-06-04 07:03:57 +00:00
|
|
|
void print_hex_data(FILE *fh, gint format, epan_dissect_t *edt)
|
2000-01-06 07:33:35 +00:00
|
|
|
{
|
2001-03-23 18:44:20 +00:00
|
|
|
gboolean multiple_sources;
|
2002-02-18 01:08:44 +00:00
|
|
|
GSList *src_le;
|
|
|
|
data_source *src;
|
2001-03-23 18:44:20 +00:00
|
|
|
tvbuff_t *tvb;
|
|
|
|
char *name;
|
|
|
|
char *line;
|
2002-08-02 23:36:07 +00:00
|
|
|
const guchar *cp;
|
2001-03-23 18:44:20 +00:00
|
|
|
guint length;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set "multiple_sources" iff this frame has more than one
|
|
|
|
* data source; if it does, we need to print the name of
|
|
|
|
* the data source before printing the data from the
|
|
|
|
* data source.
|
|
|
|
*/
|
2002-06-04 07:03:57 +00:00
|
|
|
multiple_sources = (edt->pi.data_src->next != NULL);
|
2001-03-23 18:44:20 +00:00
|
|
|
|
2002-06-04 07:03:57 +00:00
|
|
|
for (src_le = edt->pi.data_src; src_le != NULL;
|
|
|
|
src_le = src_le->next) {
|
2002-02-18 01:08:44 +00:00
|
|
|
src = src_le->data;
|
|
|
|
tvb = src->tvb;
|
2001-03-23 18:44:20 +00:00
|
|
|
if (multiple_sources) {
|
2002-02-18 01:08:44 +00:00
|
|
|
name = src->name;
|
2002-06-29 09:45:06 +00:00
|
|
|
print_line(fh, 0, format, "");
|
2002-06-22 01:24:23 +00:00
|
|
|
line = g_malloc(strlen(name) + 2); /* <name>:\0 */
|
2001-03-23 18:44:20 +00:00
|
|
|
strcpy(line, name);
|
2002-06-22 01:24:23 +00:00
|
|
|
strcat(line, ":");
|
2002-06-29 09:45:06 +00:00
|
|
|
print_line(fh, 0, format, line);
|
2001-03-23 18:44:20 +00:00
|
|
|
g_free(line);
|
|
|
|
}
|
|
|
|
length = tvb_length(tvb);
|
|
|
|
cp = tvb_get_ptr(tvb, 0, length);
|
2002-06-22 01:24:23 +00:00
|
|
|
print_hex_data_buffer(fh, cp, length,
|
|
|
|
edt->pi.fd->flags.encoding, format);
|
2001-03-23 18:44:20 +00:00
|
|
|
}
|
2000-01-06 07:33:35 +00:00
|
|
|
}
|
|
|
|
|
2002-06-21 23:52:47 +00:00
|
|
|
/*
|
|
|
|
* This routine is based on a routine created by Dan Lasley
|
|
|
|
* <DLASLEY@PROMUS.com>.
|
|
|
|
*
|
|
|
|
* It was modified for Ethereal by Gilbert Ramirez and others.
|
|
|
|
*/
|
2002-06-22 22:31:29 +00:00
|
|
|
|
|
|
|
#define MAX_OFFSET_LEN 8 /* max length of hex offset of bytes */
|
|
|
|
#define BYTES_PER_LINE 16 /* max byte values printed on a line */
|
|
|
|
#define HEX_DUMP_LEN (BYTES_PER_LINE*3)
|
|
|
|
/* max number of characters hex dump takes -
|
|
|
|
2 digits plus trailing blank */
|
|
|
|
#define DATA_DUMP_LEN (HEX_DUMP_LEN + 2 + BYTES_PER_LINE)
|
|
|
|
/* number of characters those bytes take;
|
|
|
|
3 characters per byte of hex dump,
|
|
|
|
2 blanks separating hex from ASCII,
|
|
|
|
1 character per byte of ASCII dump */
|
|
|
|
#define MAX_LINE_LEN (MAX_OFFSET_LEN + 2 + DATA_DUMP_LEN)
|
|
|
|
/* number of characters per line;
|
|
|
|
offset, 2 blanks separating offset
|
|
|
|
from data dump, data dump */
|
|
|
|
|
2002-06-22 01:24:23 +00:00
|
|
|
static void
|
2002-08-02 23:36:07 +00:00
|
|
|
print_hex_data_buffer(FILE *fh, register const guchar *cp,
|
|
|
|
register guint length, char_enc encoding, gint format)
|
1998-09-16 02:39:15 +00:00
|
|
|
{
|
2002-06-22 22:31:29 +00:00
|
|
|
register unsigned int ad, i, j, k, l;
|
2002-08-02 23:36:07 +00:00
|
|
|
guchar c;
|
|
|
|
guchar line[MAX_LINE_LEN + 1];
|
2002-06-22 22:31:29 +00:00
|
|
|
unsigned int use_digits;
|
2002-08-02 23:36:07 +00:00
|
|
|
static guchar binhex[16] = {
|
2000-01-07 00:36:25 +00:00
|
|
|
'0', '1', '2', '3', '4', '5', '6', '7',
|
|
|
|
'8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};
|
1998-09-16 02:39:15 +00:00
|
|
|
|
2002-06-29 09:45:06 +00:00
|
|
|
print_line(fh, 0, format, "");
|
2002-06-22 22:31:29 +00:00
|
|
|
|
|
|
|
/*
|
2002-06-23 23:43:32 +00:00
|
|
|
* How many of the leading digits of the offset will we supply?
|
|
|
|
* We always supply at least 4 digits, but if the maximum offset
|
|
|
|
* won't fit in 4 digits, we use as many digits as will be needed.
|
2002-06-22 22:31:29 +00:00
|
|
|
*/
|
|
|
|
if (((length - 1) & 0xF0000000) != 0)
|
|
|
|
use_digits = 8; /* need all 8 digits */
|
|
|
|
else if (((length - 1) & 0x0F000000) != 0)
|
|
|
|
use_digits = 7; /* need 7 digits */
|
|
|
|
else if (((length - 1) & 0x00F00000) != 0)
|
|
|
|
use_digits = 6; /* need 6 digits */
|
|
|
|
else if (((length - 1) & 0x000F0000) != 0)
|
|
|
|
use_digits = 5; /* need 5 digits */
|
|
|
|
else
|
|
|
|
use_digits = 4; /* we'll supply 4 digits */
|
|
|
|
|
2002-06-21 23:52:47 +00:00
|
|
|
ad = 0;
|
2002-06-22 01:24:23 +00:00
|
|
|
i = 0;
|
2002-06-21 23:52:47 +00:00
|
|
|
j = 0;
|
|
|
|
k = 0;
|
2002-06-22 01:24:23 +00:00
|
|
|
while (i < length) {
|
|
|
|
if ((i & 15) == 0) {
|
|
|
|
/*
|
|
|
|
* Start of a new line.
|
|
|
|
*/
|
|
|
|
j = 0;
|
|
|
|
k = 0;
|
2002-06-22 22:31:29 +00:00
|
|
|
l = use_digits;
|
|
|
|
do {
|
|
|
|
l--;
|
|
|
|
c = (ad >> (l*4)) & 0xF;
|
|
|
|
line[j++] = binhex[c];
|
|
|
|
} while (l != 0);
|
|
|
|
line[j++] = ' ';
|
|
|
|
line[j++] = ' ';
|
|
|
|
memset(line+j, ' ', DATA_DUMP_LEN);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Offset in line of ASCII dump.
|
|
|
|
*/
|
|
|
|
k = j + HEX_DUMP_LEN + 2;
|
2002-06-22 01:24:23 +00:00
|
|
|
}
|
2002-06-21 23:52:47 +00:00
|
|
|
c = *cp++;
|
2002-06-22 22:31:29 +00:00
|
|
|
line[j++] = binhex[c>>4];
|
|
|
|
line[j++] = binhex[c&0xf];
|
2002-06-21 23:52:47 +00:00
|
|
|
j++;
|
1999-11-22 06:24:56 +00:00
|
|
|
if (encoding == CHAR_EBCDIC) {
|
|
|
|
c = EBCDIC_to_ASCII1(c);
|
|
|
|
}
|
2002-06-22 22:31:29 +00:00
|
|
|
line[k++] = c >= ' ' && c < 0x7f ? c : '.';
|
2002-06-22 01:24:23 +00:00
|
|
|
i++;
|
|
|
|
if ((i & 15) == 0 || i == length) {
|
|
|
|
/*
|
|
|
|
* We'll be starting a new line, or
|
|
|
|
* we're finished printing this buffer;
|
|
|
|
* dump out the line we've constructed,
|
|
|
|
* and advance the offset.
|
|
|
|
*/
|
2002-06-22 22:31:29 +00:00
|
|
|
line[k] = '\0';
|
2002-06-29 09:45:06 +00:00
|
|
|
print_line(fh, 0, format, line);
|
2002-06-21 23:52:47 +00:00
|
|
|
ad += 16;
|
|
|
|
}
|
|
|
|
}
|
1998-09-16 02:39:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
void ps_clean_string(unsigned char *out, const unsigned char *in,
|
|
|
|
int outbuf_size)
|
|
|
|
{
|
|
|
|
int rd, wr;
|
|
|
|
char c;
|
|
|
|
|
|
|
|
for (rd = 0, wr = 0 ; wr < outbuf_size; rd++, wr++ ) {
|
|
|
|
c = in[rd];
|
|
|
|
switch (c) {
|
|
|
|
case '(':
|
|
|
|
case ')':
|
|
|
|
case '\\':
|
|
|
|
out[wr] = '\\';
|
|
|
|
out[++wr] = c;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
out[wr] = c;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
When printing a packet, do it from the protocol tree, not from the GTK+
tree constructed from the protocol tree:
1) The value of "level" field of GTK+ tree items appears to
depend on various random things - see a change I made to
"packet-dns.c" a while ago, to change the order in which
items were put in the tree, so that DNS trees printed with
correct indentation - and, right now, we appear to be doing
*something* wrong, as some packets I printed from one file
here had randomly bogus indentation; I could probably track
the problem down and fix it, but that might just hold us
until we accidentally do something *else* wrong by GTK+'s
lights.
The new code provides its own tree level as it goes.
2) The new code is independent of GTK+, so it could be used with
other toolkits, or with non-GUI variants of Ethereal.
3) This may make it easier to add a "Print..." menu item to let
the user print packets other than the currently selected
packet.
Make the internal routines used to print the packet static.
For the "Print Packet" menu item, put up a message box if they haven't
yet selected a packet.
svn path=/trunk/; revision=362
1999-07-13 04:38:15 +00:00
|
|
|
|
2002-06-29 09:45:06 +00:00
|
|
|
void print_preamble(FILE *fh, gint format)
|
|
|
|
{
|
|
|
|
if (format == PR_FMT_PS)
|
|
|
|
print_ps_preamble(fh);
|
|
|
|
}
|
|
|
|
|
|
|
|
void print_finale(FILE *fh, gint format)
|
|
|
|
{
|
|
|
|
if (format == PR_FMT_PS)
|
|
|
|
print_ps_finale(fh);
|
|
|
|
}
|
|
|
|
|
|
|
|
void print_line(FILE *fh, int indent, gint format, char *line)
|
2000-01-06 07:33:35 +00:00
|
|
|
{
|
2002-06-29 09:45:06 +00:00
|
|
|
char space[MAX_INDENT+1];
|
|
|
|
int i;
|
|
|
|
int num_spaces;
|
2002-06-22 01:43:57 +00:00
|
|
|
char psbuffer[MAX_PS_LINE_LENGTH]; /* static sized buffer! */
|
2000-01-06 07:33:35 +00:00
|
|
|
|
|
|
|
if (format == PR_FMT_PS) {
|
2002-06-22 01:43:57 +00:00
|
|
|
ps_clean_string(psbuffer, line, MAX_PS_LINE_LENGTH);
|
2002-06-29 09:45:06 +00:00
|
|
|
fprintf(fh, "%d (%s) putline\n", indent, psbuffer);
|
2002-06-22 01:24:23 +00:00
|
|
|
} else {
|
2002-06-29 09:45:06 +00:00
|
|
|
/* Prepare the tabs for printing, depending on tree level */
|
|
|
|
num_spaces = indent * 4;
|
|
|
|
if (num_spaces > MAX_INDENT) {
|
|
|
|
num_spaces = MAX_INDENT;
|
|
|
|
}
|
|
|
|
for (i = 0; i < num_spaces; i++) {
|
|
|
|
space[i] = ' ';
|
|
|
|
}
|
|
|
|
/* The string is NUL-terminated */
|
|
|
|
space[num_spaces] = '\0';
|
|
|
|
|
|
|
|
fputs(space, fh);
|
2000-01-06 07:33:35 +00:00
|
|
|
fputs(line, fh);
|
2002-06-22 01:24:23 +00:00
|
|
|
putc('\n', fh);
|
|
|
|
}
|
2000-01-06 07:33:35 +00:00
|
|
|
}
|