2009-02-15 21:47:57 +00:00
|
|
|
/* dftest.c
|
|
|
|
* Shows display filter byte-code, for debugging dfilter routines.
|
2001-02-01 20:21:25 +00:00
|
|
|
*
|
2006-05-21 05:12:17 +00:00
|
|
|
* Wireshark - Network traffic analyzer
|
|
|
|
* By Gerald Combs <gerald@wireshark.org>
|
2001-02-01 20:21:25 +00:00
|
|
|
* Copyright 1998 Gerald Combs
|
|
|
|
*
|
2018-02-07 12:26:45 +01:00
|
|
|
* SPDX-License-Identifier: GPL-2.0-or-later
|
2001-02-01 20:21:25 +00:00
|
|
|
*/
|
|
|
|
|
2014-08-22 22:13:05 +01:00
|
|
|
#include <config.h>
|
2022-10-06 18:41:17 +01:00
|
|
|
#define WS_LOG_DOMAIN LOG_DOMAIN_MAIN
|
2001-02-01 20:21:25 +00:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <locale.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include <glib.h>
|
|
|
|
|
2023-02-06 22:16:37 +00:00
|
|
|
#include <ws_exit_codes.h>
|
|
|
|
|
2013-12-02 08:30:29 +00:00
|
|
|
#include <epan/epan.h>
|
2002-01-21 07:37:49 +00:00
|
|
|
#include <epan/timestamp.h>
|
2013-12-02 08:30:29 +00:00
|
|
|
#include <epan/prefs.h>
|
|
|
|
#include <epan/dfilter/dfilter.h>
|
2023-11-08 09:43:29 +00:00
|
|
|
#include <epan/dfilter/dfilter-macro.h>
|
2013-12-02 08:30:29 +00:00
|
|
|
|
2014-12-01 23:08:50 -05:00
|
|
|
#ifdef HAVE_PLUGINS
|
2013-12-02 08:30:29 +00:00
|
|
|
#include <wsutil/plugins.h>
|
2014-12-01 23:08:50 -05:00
|
|
|
#endif
|
2013-11-17 02:55:14 +00:00
|
|
|
#include <wsutil/filesystem.h>
|
2024-10-16 20:29:04 -04:00
|
|
|
#include <wsutil/file_util.h>
|
2008-06-30 17:16:29 +00:00
|
|
|
#include <wsutil/privileges.h>
|
2025-04-28 13:30:46 +00:00
|
|
|
#include <wsutil/clopts_common.h>
|
2021-06-19 19:44:58 +01:00
|
|
|
#include <wsutil/wslog.h>
|
2022-12-30 16:11:53 +00:00
|
|
|
#include <wsutil/ws_getopt.h>
|
2023-11-08 09:43:29 +00:00
|
|
|
#include <wsutil/utf8_entities.h>
|
2013-12-02 08:30:29 +00:00
|
|
|
|
2021-12-31 16:39:38 +00:00
|
|
|
#include <wiretap/wtap.h>
|
|
|
|
|
2012-02-17 12:30:27 +00:00
|
|
|
#include "ui/util.h"
|
2023-02-06 22:46:45 +00:00
|
|
|
#include "wsutil/cmdarg_err.h"
|
2021-03-15 11:29:43 -07:00
|
|
|
#include "ui/failure_message.h"
|
2023-02-06 21:57:51 +00:00
|
|
|
#include "wsutil/version_info.h"
|
2025-05-25 19:31:07 -04:00
|
|
|
#include "cli_main.h"
|
2001-02-01 20:21:25 +00:00
|
|
|
|
2024-04-02 14:37:13 +02:00
|
|
|
static int opt_verbose;
|
|
|
|
static int opt_debug_level; /* currently up to 2 */
|
|
|
|
static int opt_flex;
|
|
|
|
static int opt_lemon;
|
|
|
|
static int opt_syntax_tree;
|
|
|
|
static int opt_return_vals;
|
|
|
|
static int opt_timer;
|
2025-04-21 12:15:22 -04:00
|
|
|
static int opt_optimize = 1;
|
2024-04-02 14:37:13 +02:00
|
|
|
static int opt_show_types;
|
|
|
|
static int opt_dump_refs;
|
|
|
|
static int opt_dump_macros;
|
2023-01-07 02:40:14 +00:00
|
|
|
|
2024-07-07 08:44:20 -04:00
|
|
|
static int64_t elapsed_expand;
|
|
|
|
static int64_t elapsed_compile;
|
2023-01-07 02:40:14 +00:00
|
|
|
|
2024-10-16 20:29:04 -04:00
|
|
|
#ifndef HAVE_GETLINE
|
|
|
|
/* Maximum supported line length of a filter. */
|
|
|
|
#define MAX_LINELEN 4096
|
|
|
|
|
|
|
|
/** Read a line without trailing (CR)LF. Returns -1 on failure. */
|
|
|
|
static int
|
|
|
|
fgetline(char *buf, int size, FILE *fp)
|
|
|
|
{
|
|
|
|
if (fgets(buf, size, fp)) {
|
|
|
|
int len = (int)strcspn(buf, "\r\n");
|
|
|
|
buf[len] = '\0';
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
} /* fgetline */
|
|
|
|
#endif /* HAVE_GETLINE */
|
|
|
|
|
2022-04-07 13:57:39 +01:00
|
|
|
static void
|
2022-12-22 10:44:33 +00:00
|
|
|
putloc(FILE *fp, df_loc_t loc)
|
2022-04-07 13:57:39 +01:00
|
|
|
{
|
2022-04-11 21:52:53 +01:00
|
|
|
for (long i = 0; i < loc.col_start; i++) {
|
2022-04-07 13:57:39 +01:00
|
|
|
fputc(' ', fp);
|
|
|
|
}
|
|
|
|
fputc('^', fp);
|
|
|
|
|
2022-04-11 21:52:53 +01:00
|
|
|
for (size_t l = loc.col_len; l > 1; l--) {
|
2022-04-07 13:57:39 +01:00
|
|
|
fputc('~', fp);
|
|
|
|
}
|
|
|
|
fputc('\n', fp);
|
|
|
|
}
|
|
|
|
|
2025-04-21 12:15:22 -04:00
|
|
|
static void
|
|
|
|
print_usage(void)
|
2022-12-30 16:11:53 +00:00
|
|
|
{
|
2023-01-07 02:40:14 +00:00
|
|
|
FILE *fp = stdout;
|
2023-01-08 17:00:00 +00:00
|
|
|
fprintf(fp, "\n");
|
2023-01-07 02:40:14 +00:00
|
|
|
fprintf(fp, "Usage: dftest [OPTIONS] -- EXPRESSION\n");
|
|
|
|
fprintf(fp, "Options:\n");
|
|
|
|
fprintf(fp, " -V, --verbose enable verbose mode\n");
|
2024-10-11 18:10:58 -04:00
|
|
|
fprintf(fp, " -C <config profile> run with specified configuration profile\n");
|
2023-10-22 13:45:01 +01:00
|
|
|
fprintf(fp, " -d, --debug[=N] increase or set debug level\n");
|
|
|
|
fprintf(fp, " -D set maximum debug level\n");
|
2023-01-07 02:40:14 +00:00
|
|
|
fprintf(fp, " -f, --flex enable Flex debug trace\n");
|
|
|
|
fprintf(fp, " -l, --lemon enable Lemon debug trace\n");
|
|
|
|
fprintf(fp, " -s, --syntax print syntax tree\n");
|
2023-11-08 09:43:29 +00:00
|
|
|
fprintf(fp, " -m --macros print saved macros\n");
|
2023-01-07 02:40:14 +00:00
|
|
|
fprintf(fp, " -t, --timer print elapsed compilation time\n");
|
2024-02-12 20:23:37 -05:00
|
|
|
fprintf(fp, " -r --return-vals return field values for the tree root\n");
|
2023-01-08 17:00:00 +00:00
|
|
|
fprintf(fp, " -0, --optimize=0 do not optimize (check syntax)\n");
|
2023-01-10 15:42:32 +00:00
|
|
|
fprintf(fp, " --types show field value types\n");
|
2023-04-17 01:52:16 +01:00
|
|
|
/* NOTE: References are loaded during runtime and dftest only does compilation.
|
|
|
|
* Unless some static reference data is hard-coded at compile time during
|
|
|
|
* development the --refs option to dftest is useless because it will just
|
|
|
|
* print empty reference vectors. */
|
|
|
|
fprintf(fp, " --refs dump some runtime data structures\n");
|
2024-10-16 20:29:04 -04:00
|
|
|
fprintf(fp, " --file <path> read filters line-by-line from a file (use '-' for stdin)\n");
|
2023-01-07 02:40:14 +00:00
|
|
|
fprintf(fp, " -h, --help display this help and exit\n");
|
|
|
|
fprintf(fp, " -v, --version print version\n");
|
2023-01-09 03:12:49 +00:00
|
|
|
fprintf(fp, "\n");
|
|
|
|
ws_log_print_usage(fp);
|
2023-01-05 17:37:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_syntax_tree(dfilter_t *df)
|
|
|
|
{
|
|
|
|
printf("Syntax tree:\n%s\n\n", dfilter_syntax_tree(df));
|
|
|
|
}
|
|
|
|
|
2023-11-08 09:43:29 +00:00
|
|
|
static void
|
|
|
|
print_macros(void)
|
|
|
|
{
|
|
|
|
if (dfilter_macro_table_count() == 0) {
|
|
|
|
printf("Macros: (empty)\n\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct dfilter_macro_table_iter iter;
|
|
|
|
const char *name, *text;
|
|
|
|
|
|
|
|
dfilter_macro_table_iter_init(&iter);
|
|
|
|
printf("Macros:\n");
|
|
|
|
while (dfilter_macro_table_iter_next(&iter, &name, &text)) {
|
|
|
|
printf(" "UTF8_BULLET" %s:\n", name);
|
|
|
|
printf(" %s\n", text);
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
2023-01-05 17:37:59 +00:00
|
|
|
static void
|
|
|
|
print_warnings(dfilter_t *df)
|
|
|
|
{
|
2024-07-07 08:44:20 -04:00
|
|
|
unsigned i;
|
2023-01-05 17:37:59 +00:00
|
|
|
GPtrArray *deprecated;
|
2023-01-07 02:40:14 +00:00
|
|
|
int count = 0;
|
2023-01-05 17:37:59 +00:00
|
|
|
|
2023-01-07 02:40:14 +00:00
|
|
|
for (GSList *l = dfilter_get_warnings(df); l != NULL; l = l->next) {
|
|
|
|
printf("\nWarning: %s.", (char *)l->data);
|
|
|
|
count++;
|
2023-01-05 17:37:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
deprecated = dfilter_deprecated_tokens(df);
|
|
|
|
if (deprecated && deprecated->len) {
|
|
|
|
for (i = 0; i < deprecated->len; i++) {
|
2023-01-07 02:40:14 +00:00
|
|
|
const char *token = g_ptr_array_index(deprecated, i);
|
|
|
|
printf("\nWarning: Deprecated token \"%s\".", token);
|
|
|
|
count++;
|
2023-01-05 17:37:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-07 02:40:14 +00:00
|
|
|
if (count) {
|
2023-01-05 17:37:59 +00:00
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2023-01-07 02:40:14 +00:00
|
|
|
print_elapsed(void)
|
|
|
|
{
|
2023-08-26 02:20:15 +01:00
|
|
|
printf("\nElapsed: %"PRId64" µs (%"PRId64" µs + %"PRId64" µs)\n",
|
|
|
|
elapsed_expand + elapsed_compile,
|
|
|
|
elapsed_expand,
|
|
|
|
elapsed_compile);
|
2023-01-07 02:40:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static char *
|
2023-08-26 02:20:15 +01:00
|
|
|
expand_filter(const char *text)
|
2023-01-05 17:37:59 +00:00
|
|
|
{
|
2023-01-07 02:40:14 +00:00
|
|
|
char *expanded = NULL;
|
2023-04-15 00:18:53 +01:00
|
|
|
df_error_t *err = NULL;
|
2024-07-07 08:44:20 -04:00
|
|
|
int64_t start;
|
2023-01-07 02:40:14 +00:00
|
|
|
|
2023-08-26 02:20:15 +01:00
|
|
|
start = g_get_monotonic_time();
|
2023-04-15 00:18:53 +01:00
|
|
|
expanded = dfilter_expand(text, &err);
|
2023-01-07 02:40:14 +00:00
|
|
|
if (expanded == NULL) {
|
2023-04-15 00:18:53 +01:00
|
|
|
fprintf(stderr, "Error: %s\n", err->msg);
|
|
|
|
df_error_free(&err);
|
2023-01-07 02:40:14 +00:00
|
|
|
}
|
2023-08-26 02:20:15 +01:00
|
|
|
elapsed_expand = g_get_monotonic_time() - start;
|
2023-01-07 02:40:14 +00:00
|
|
|
return expanded;
|
|
|
|
}
|
|
|
|
|
2024-07-07 08:44:20 -04:00
|
|
|
static bool
|
2023-08-26 02:20:15 +01:00
|
|
|
compile_filter(const char *text, dfilter_t **dfp)
|
2023-01-07 02:40:14 +00:00
|
|
|
{
|
|
|
|
unsigned df_flags = 0;
|
2024-07-07 08:44:20 -04:00
|
|
|
bool ok;
|
2023-01-07 02:40:14 +00:00
|
|
|
df_error_t *df_err = NULL;
|
2024-07-07 08:44:20 -04:00
|
|
|
int64_t start;
|
2023-01-07 02:40:14 +00:00
|
|
|
|
2023-01-08 17:00:00 +00:00
|
|
|
if (opt_optimize > 0)
|
|
|
|
df_flags |= DF_OPTIMIZE;
|
2023-01-07 02:40:14 +00:00
|
|
|
if (opt_syntax_tree)
|
|
|
|
df_flags |= DF_SAVE_TREE;
|
|
|
|
if (opt_flex)
|
|
|
|
df_flags |= DF_DEBUG_FLEX;
|
|
|
|
if (opt_lemon)
|
|
|
|
df_flags |= DF_DEBUG_LEMON;
|
2024-02-12 20:23:37 -05:00
|
|
|
if (opt_return_vals)
|
|
|
|
df_flags |= DF_RETURN_VALUES;
|
2023-01-07 02:40:14 +00:00
|
|
|
|
2023-08-26 02:20:15 +01:00
|
|
|
start = g_get_monotonic_time();
|
2023-04-21 00:18:36 +01:00
|
|
|
ok = dfilter_compile_full(text, dfp, &df_err, df_flags, "dftest");
|
2023-01-07 02:40:14 +00:00
|
|
|
if (!ok) {
|
|
|
|
fprintf(stderr, "Error: %s\n", df_err->msg);
|
|
|
|
if (df_err->loc.col_start >= 0) {
|
|
|
|
fprintf(stderr, " %s\n ", text);
|
|
|
|
putloc(stderr, df_err->loc);
|
|
|
|
}
|
2023-04-15 00:18:53 +01:00
|
|
|
df_error_free(&df_err);
|
2023-01-07 02:40:14 +00:00
|
|
|
}
|
2023-08-26 02:20:15 +01:00
|
|
|
elapsed_compile = g_get_monotonic_time() - start;
|
2023-01-07 02:40:14 +00:00
|
|
|
return ok;
|
2022-12-30 16:11:53 +00:00
|
|
|
}
|
|
|
|
|
2025-04-21 12:15:22 -04:00
|
|
|
static bool
|
|
|
|
optarg_to_digit(const char *arg, int* digit)
|
2023-10-22 13:45:01 +01:00
|
|
|
{
|
|
|
|
if (strlen(arg) > 1 || !g_ascii_isdigit(*arg)) {
|
|
|
|
printf("Error: \"%s\" is not a valid number 0-9\n", arg);
|
2025-04-21 12:15:22 -04:00
|
|
|
print_usage();
|
|
|
|
return false;
|
2023-10-22 13:45:01 +01:00
|
|
|
}
|
|
|
|
errno = 0;
|
2025-04-21 12:15:22 -04:00
|
|
|
*digit = (int)strtol(ws_optarg, NULL, 10);
|
2023-10-22 13:45:01 +01:00
|
|
|
if (errno) {
|
|
|
|
printf("Error: %s\n", g_strerror(errno));
|
2025-04-21 12:15:22 -04:00
|
|
|
print_usage();
|
|
|
|
return false;
|
2023-10-22 13:45:01 +01:00
|
|
|
}
|
2025-04-21 12:15:22 -04:00
|
|
|
return true;
|
2023-10-22 13:45:01 +01:00
|
|
|
}
|
|
|
|
|
2024-10-16 20:29:04 -04:00
|
|
|
static int
|
|
|
|
test_filter(const char *text)
|
|
|
|
{
|
|
|
|
char *expanded_text = NULL;
|
|
|
|
dfilter_t *df = NULL;
|
|
|
|
|
|
|
|
printf("Filter:\n %s\n\n", text);
|
|
|
|
|
|
|
|
/* Expand macros. */
|
|
|
|
expanded_text = expand_filter(text);
|
|
|
|
if (expanded_text == NULL) {
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(text, expanded_text) != 0)
|
|
|
|
printf("Filter (after expansion):\n %s\n\n", expanded_text);
|
|
|
|
|
|
|
|
/* Compile it */
|
|
|
|
if (!compile_filter(expanded_text, &df)) {
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If logging is enabled add an empty line. */
|
|
|
|
if (opt_debug_level > 0) {
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (df == NULL) {
|
|
|
|
printf("Filter is empty.\n");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt_syntax_tree)
|
|
|
|
print_syntax_tree(df);
|
|
|
|
|
|
|
|
uint16_t dump_flags = 0;
|
|
|
|
if (opt_show_types)
|
|
|
|
dump_flags |= DF_DUMP_SHOW_FTYPE;
|
|
|
|
if (opt_dump_refs)
|
|
|
|
dump_flags |= DF_DUMP_REFERENCES;
|
|
|
|
|
|
|
|
dfilter_dump(stdout, df, dump_flags);
|
|
|
|
|
|
|
|
print_warnings(df);
|
|
|
|
|
|
|
|
if (opt_timer)
|
|
|
|
print_elapsed();
|
|
|
|
|
|
|
|
g_free(expanded_text);
|
|
|
|
dfilter_free(df);
|
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
|
|
|
|
fail:
|
|
|
|
g_free(expanded_text);
|
|
|
|
dfilter_free(df);
|
|
|
|
return WS_EXIT_INVALID_FILTER;
|
|
|
|
}
|
|
|
|
|
2001-02-01 20:21:25 +00:00
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
2022-02-20 12:09:34 -08:00
|
|
|
char *configuration_init_error;
|
2024-10-16 20:29:04 -04:00
|
|
|
char *path = NULL;
|
2022-12-29 20:06:19 +00:00
|
|
|
char *text = NULL;
|
2023-01-07 02:40:14 +00:00
|
|
|
int exit_status = EXIT_FAILURE;
|
2022-10-06 18:41:17 +01:00
|
|
|
|
2025-04-28 13:30:46 +00:00
|
|
|
const char* optstring = "hvC:dDflsmrtV0";
|
|
|
|
static const struct ws_option long_options[] = {
|
|
|
|
{ "help", ws_no_argument, 0, 'h' },
|
|
|
|
{ "version", ws_no_argument, 0, 'v' },
|
|
|
|
{ "debug", ws_optional_argument, 0, 'd' },
|
|
|
|
{ "flex", ws_no_argument, 0, 'f' },
|
|
|
|
{ "lemon", ws_no_argument, 0, 'l' },
|
|
|
|
{ "syntax", ws_no_argument, 0, 's' },
|
|
|
|
{ "macros", ws_no_argument, 0, 'm' },
|
|
|
|
{ "timer", ws_no_argument, 0, 't' },
|
|
|
|
{ "verbose", ws_no_argument, 0, 'V' },
|
|
|
|
{ "return-vals", ws_no_argument, 0, 'r' },
|
|
|
|
{ "optimize", ws_required_argument, 0, 1000 },
|
|
|
|
{ "types", ws_no_argument, 0, 2000 },
|
|
|
|
{ "refs", ws_no_argument, 0, 3000 },
|
|
|
|
{ "file", ws_required_argument, 0, 4000 },
|
|
|
|
LONGOPT_WSLOG
|
|
|
|
{ NULL, 0, 0, 0 }
|
|
|
|
};
|
|
|
|
int opt;
|
|
|
|
|
2024-11-28 14:04:14 -08:00
|
|
|
/* Set the program name. */
|
|
|
|
g_set_prgname("dftest");
|
|
|
|
|
2022-12-30 04:00:22 +00:00
|
|
|
/*
|
|
|
|
* Set the C-language locale to the native environment and set the
|
|
|
|
* code page to UTF-8 on Windows.
|
|
|
|
*/
|
|
|
|
#ifdef _WIN32
|
|
|
|
setlocale(LC_ALL, ".UTF-8");
|
|
|
|
#else
|
|
|
|
setlocale(LC_ALL, "");
|
|
|
|
#endif
|
|
|
|
|
2024-11-28 14:04:14 -08:00
|
|
|
cmdarg_err_init(stderr_cmdarg_err, stderr_cmdarg_err_cont);
|
2023-01-07 02:40:14 +00:00
|
|
|
|
|
|
|
/* Initialize log handler early for startup. */
|
2024-11-28 14:04:14 -08:00
|
|
|
ws_log_init(vcmdarg_err);
|
2023-01-07 02:40:14 +00:00
|
|
|
|
|
|
|
/* Early logging command-line initialization. */
|
2025-04-28 13:30:46 +00:00
|
|
|
ws_log_parse_args(&argc, argv, optstring, long_options, vcmdarg_err, WS_EXIT_INVALID_OPTION);
|
2023-01-07 02:40:14 +00:00
|
|
|
|
|
|
|
ws_noisy("Finished log init and parsing command line log arguments");
|
|
|
|
|
2024-10-11 18:10:58 -04:00
|
|
|
/*
|
|
|
|
* Get credential information for later use.
|
|
|
|
*/
|
|
|
|
init_process_policies();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Attempt to get the pathname of the directory containing the
|
|
|
|
* executable file.
|
|
|
|
*/
|
2024-11-21 09:45:22 -08:00
|
|
|
configuration_init_error = configuration_init(argv[0]);
|
2024-10-11 18:10:58 -04:00
|
|
|
if (configuration_init_error != NULL) {
|
|
|
|
fprintf(stderr, "Error: Can't get pathname of directory containing "
|
|
|
|
"the dftest program: %s.\n",
|
|
|
|
configuration_init_error);
|
|
|
|
g_free(configuration_init_error);
|
|
|
|
}
|
|
|
|
|
2023-01-06 00:33:08 +00:00
|
|
|
ws_init_version_info("DFTest", NULL, NULL);
|
|
|
|
|
2023-01-07 02:40:14 +00:00
|
|
|
for (;;) {
|
|
|
|
opt = ws_getopt_long(argc, argv, optstring, long_options, NULL);
|
|
|
|
if (opt == -1)
|
|
|
|
break;
|
|
|
|
|
2022-12-30 16:11:53 +00:00
|
|
|
switch (opt) {
|
2023-01-07 02:40:14 +00:00
|
|
|
case 'V':
|
2023-01-05 17:37:59 +00:00
|
|
|
opt_verbose = 1;
|
|
|
|
break;
|
2022-12-30 16:11:53 +00:00
|
|
|
case 'd':
|
2023-10-22 13:45:01 +01:00
|
|
|
if (ws_optarg) {
|
2025-04-21 12:15:22 -04:00
|
|
|
if (!optarg_to_digit(ws_optarg, &opt_debug_level))
|
|
|
|
return WS_EXIT_INVALID_OPTION;
|
2023-10-22 13:45:01 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
opt_debug_level++;
|
|
|
|
}
|
|
|
|
opt_show_types = 1;
|
|
|
|
break;
|
2024-10-11 18:10:58 -04:00
|
|
|
case 'C': /* Configuration Profile */
|
|
|
|
if (profile_exists (ws_optarg, false)) {
|
|
|
|
set_profile_name (ws_optarg);
|
|
|
|
} else {
|
|
|
|
cmdarg_err("Configuration Profile \"%s\" does not exist", ws_optarg);
|
2025-04-21 12:15:22 -04:00
|
|
|
print_usage();
|
|
|
|
return WS_EXIT_INVALID_OPTION;
|
2024-10-11 18:10:58 -04:00
|
|
|
}
|
|
|
|
break;
|
2023-10-22 13:45:01 +01:00
|
|
|
case 'D':
|
|
|
|
opt_debug_level = 9;
|
2023-11-03 17:22:54 +00:00
|
|
|
opt_lemon = 1;
|
|
|
|
opt_flex = 1;
|
2023-10-03 16:31:05 +01:00
|
|
|
opt_show_types = 1;
|
2022-12-30 16:11:53 +00:00
|
|
|
break;
|
|
|
|
case 'f':
|
2023-01-05 17:37:59 +00:00
|
|
|
opt_flex = 1;
|
2022-12-30 16:11:53 +00:00
|
|
|
break;
|
|
|
|
case 'l':
|
2023-01-05 17:37:59 +00:00
|
|
|
opt_lemon = 1;
|
|
|
|
break;
|
2023-01-06 00:33:08 +00:00
|
|
|
case 's':
|
2023-01-05 17:37:59 +00:00
|
|
|
opt_syntax_tree = 1;
|
2022-12-30 16:11:53 +00:00
|
|
|
break;
|
2023-11-08 09:43:29 +00:00
|
|
|
case 'm':
|
|
|
|
opt_dump_macros = 1;
|
|
|
|
break;
|
2023-01-07 02:40:14 +00:00
|
|
|
case 't':
|
|
|
|
opt_timer = 1;
|
|
|
|
break;
|
2024-02-12 20:23:37 -05:00
|
|
|
case 'r':
|
|
|
|
opt_return_vals = 1;
|
|
|
|
break;
|
2023-01-08 17:00:00 +00:00
|
|
|
case '0':
|
|
|
|
opt_optimize = 0;
|
|
|
|
break;
|
|
|
|
case 1000:
|
2025-04-21 12:15:22 -04:00
|
|
|
if (!optarg_to_digit(ws_optarg, &opt_optimize))
|
|
|
|
return WS_EXIT_INVALID_OPTION;
|
2023-01-08 17:00:00 +00:00
|
|
|
break;
|
2023-01-10 15:42:32 +00:00
|
|
|
case 2000:
|
|
|
|
opt_show_types = 1;
|
|
|
|
break;
|
2023-04-17 01:52:16 +01:00
|
|
|
case 3000:
|
|
|
|
opt_dump_refs = 1;
|
|
|
|
break;
|
2024-10-16 20:29:04 -04:00
|
|
|
case 4000:
|
|
|
|
path = ws_optarg;
|
|
|
|
break;
|
2023-01-07 02:40:14 +00:00
|
|
|
case 'v':
|
2023-10-12 18:54:13 -04:00
|
|
|
show_version();
|
2025-04-21 12:15:22 -04:00
|
|
|
return EXIT_SUCCESS;
|
2023-01-06 00:33:08 +00:00
|
|
|
case 'h':
|
|
|
|
show_help_header(NULL);
|
2025-04-21 12:15:22 -04:00
|
|
|
print_usage();
|
|
|
|
return EXIT_SUCCESS;
|
2023-01-07 02:40:14 +00:00
|
|
|
case '?':
|
2025-04-21 12:15:22 -04:00
|
|
|
print_usage();
|
|
|
|
return EXIT_FAILURE;
|
2025-04-29 14:00:41 -04:00
|
|
|
break;
|
2023-01-07 02:40:14 +00:00
|
|
|
default:
|
2025-04-29 14:00:41 -04:00
|
|
|
/* wslog arguments are okay */
|
|
|
|
if (ws_log_is_wslog_arg(opt))
|
|
|
|
break;
|
|
|
|
|
2023-01-07 02:40:14 +00:00
|
|
|
ws_assert_not_reached();
|
2025-04-29 14:00:41 -04:00
|
|
|
break;
|
2022-12-30 16:11:53 +00:00
|
|
|
}
|
2022-12-30 04:00:22 +00:00
|
|
|
}
|
|
|
|
|
2023-11-08 09:43:29 +00:00
|
|
|
/* Check for filter on command line. */
|
2023-01-07 02:40:14 +00:00
|
|
|
if (argv[ws_optind] == NULL) {
|
2023-11-08 09:43:29 +00:00
|
|
|
/* If not printing macros we need a filter expression to compile. */
|
2024-10-16 20:29:04 -04:00
|
|
|
if (!opt_dump_macros && !path) {
|
2023-11-08 09:43:29 +00:00
|
|
|
printf("Error: Missing argument.\n");
|
2025-04-21 12:15:22 -04:00
|
|
|
print_usage();
|
|
|
|
return EXIT_FAILURE;
|
2023-11-08 09:43:29 +00:00
|
|
|
}
|
2023-01-07 02:40:14 +00:00
|
|
|
}
|
|
|
|
|
2023-10-22 13:45:01 +01:00
|
|
|
/* Set dfilter domain logging. */
|
|
|
|
if (opt_debug_level > 1) {
|
2023-01-05 17:37:59 +00:00
|
|
|
ws_log_set_noisy_filter(LOG_DOMAIN_DFILTER);
|
2023-01-09 02:41:07 +00:00
|
|
|
}
|
2023-10-22 13:45:01 +01:00
|
|
|
else if (opt_debug_level > 0 || opt_flex || opt_lemon) {
|
|
|
|
/* Also enable some dfilter logs with flex/lemon traces for context. */
|
2023-01-08 17:47:07 +00:00
|
|
|
ws_log_set_debug_filter(LOG_DOMAIN_DFILTER);
|
2023-01-09 02:41:07 +00:00
|
|
|
}
|
2022-12-30 04:00:22 +00:00
|
|
|
|
2024-10-05 14:01:25 -07:00
|
|
|
init_report_failure_message("dftest");
|
2022-02-20 19:39:37 +00:00
|
|
|
|
|
|
|
timestamp_set_type(TS_RELATIVE);
|
|
|
|
timestamp_set_seconds_type(TS_SECONDS_DEFAULT);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Libwiretap must be initialized before libwireshark is, so that
|
|
|
|
* dissection-time handlers for file-type-dependent blocks can
|
|
|
|
* register using the file type/subtype value for the file type.
|
|
|
|
*/
|
2024-07-07 08:44:20 -04:00
|
|
|
wtap_init(true);
|
2022-02-20 19:39:37 +00:00
|
|
|
|
|
|
|
/* Register all dissectors; we must do this before checking for the
|
|
|
|
"-g" flag, as the "-g" flag dumps a list of fields registered
|
|
|
|
by the dissectors, and we must do it before we read the preferences,
|
|
|
|
in case any dissectors register preferences. */
|
2024-07-07 08:44:20 -04:00
|
|
|
if (!epan_init(NULL, NULL, true))
|
2023-01-07 02:40:14 +00:00
|
|
|
goto out;
|
2022-02-20 19:39:37 +00:00
|
|
|
|
|
|
|
/* Load libwireshark settings from the current profile. */
|
|
|
|
epan_load_settings();
|
|
|
|
|
|
|
|
/* notify all registered modules that have had any of their preferences
|
|
|
|
changed either from one of the preferences file or from the command
|
|
|
|
line that its preferences have changed. */
|
|
|
|
prefs_apply_all();
|
|
|
|
|
2023-11-08 09:43:29 +00:00
|
|
|
if (opt_dump_macros) {
|
|
|
|
print_macros();
|
|
|
|
if (argv[ws_optind] == NULL) {
|
|
|
|
/* No filter expression, we're done. */
|
2025-04-21 12:15:22 -04:00
|
|
|
return EXIT_SUCCESS;
|
2023-11-08 09:43:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-30 16:11:53 +00:00
|
|
|
/* This is useful to prevent confusion with option parsing.
|
|
|
|
* Skips printing options and argv[0]. */
|
2023-01-05 17:37:59 +00:00
|
|
|
if (opt_verbose) {
|
|
|
|
for (int i = ws_optind; i < argc; i++) {
|
2023-01-07 02:40:14 +00:00
|
|
|
fprintf(stderr, "argv[%d]: %s\n", i, argv[i]);
|
2023-01-05 17:37:59 +00:00
|
|
|
}
|
2023-01-07 02:40:14 +00:00
|
|
|
fprintf(stderr, "\n");
|
2022-12-30 16:11:53 +00:00
|
|
|
}
|
2022-12-29 04:29:01 +00:00
|
|
|
|
2024-10-16 20:29:04 -04:00
|
|
|
if (path) {
|
|
|
|
FILE *filter_p;
|
|
|
|
if (strcmp(path, "-") == 0) {
|
|
|
|
filter_p = stdin;
|
|
|
|
} else {
|
|
|
|
filter_p = ws_fopen(path, "r");
|
|
|
|
if (filter_p == NULL) {
|
|
|
|
open_failure_message(path, errno, false);
|
|
|
|
exit_status = WS_EXIT_INVALID_FILE;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bool first = true;
|
|
|
|
#ifdef HAVE_GETLINE
|
|
|
|
char *line = NULL;
|
|
|
|
size_t len = 0;
|
|
|
|
while (getline(&line, &len, filter_p) >= 0) {
|
|
|
|
#else
|
|
|
|
char line[MAX_LINELEN];
|
|
|
|
while (fgetline(line, sizeof(line), filter_p) >= 0) {
|
|
|
|
#endif
|
|
|
|
if (first) {
|
|
|
|
first = false;
|
|
|
|
} else {
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
exit_status = test_filter(line);
|
|
|
|
/* A keep going option could be added. */
|
|
|
|
if (exit_status != EXIT_SUCCESS)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#ifdef HAVE_GETLINE
|
|
|
|
g_free(line);
|
|
|
|
#endif
|
|
|
|
fclose(filter_p);
|
|
|
|
} else {
|
2022-04-07 13:57:39 +01:00
|
|
|
|
2024-10-16 20:29:04 -04:00
|
|
|
/* Check again for filter on command line */
|
2025-04-21 12:15:22 -04:00
|
|
|
if (argv[ws_optind] != NULL) {
|
|
|
|
/* Get filter text */
|
|
|
|
text = get_args_as_string(argc, argv, ws_optind);
|
|
|
|
|
|
|
|
exit_status = test_filter(text);
|
|
|
|
} else {
|
2024-10-16 20:29:04 -04:00
|
|
|
printf("Error: Missing argument.\n");
|
2025-04-21 12:15:22 -04:00
|
|
|
print_usage();
|
|
|
|
exit_status = EXIT_FAILURE;
|
2024-10-16 20:29:04 -04:00
|
|
|
}
|
2022-04-04 23:42:52 +01:00
|
|
|
}
|
2022-12-29 20:06:19 +00:00
|
|
|
|
|
|
|
out:
|
2022-02-20 19:39:37 +00:00
|
|
|
epan_cleanup();
|
2023-01-07 02:40:14 +00:00
|
|
|
g_free(text);
|
2025-04-21 12:15:22 -04:00
|
|
|
return exit_status;
|
2001-02-01 20:21:25 +00:00
|
|
|
}
|