Stephen Hemminger 591a9d7985 add FILE argument to debug functions
The DPDK dump functions are useful for remote debugging of an
applications. But when application runs as a daemon, stdout
is typically routed to /dev/null.

Instead change all these functions to take a stdio FILE * handle
instead. An application can then use open_memstream() to capture
the output.

Signed-off-by: Stephen Hemminger <stephen@networkplumber.org>
[Thomas: fix quota_watermark example]
Acked-by: Thomas Monjalon <thomas.monjalon@6wind.com>
2014-05-16 16:02:55 +02:00

218 lines
6.3 KiB
C

/*-
* BSD LICENSE
*
* Copyright(c) 2010-2014 Intel Corporation. All rights reserved.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Intel Corporation nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include <stdio.h>
#include <stdint.h>
#include <string.h>
#include <termios.h>
#include <cmdline_rdline.h>
#include <cmdline_parse.h>
#include <cmdline_parse_num.h>
#include <cmdline_parse_string.h>
#include <cmdline.h>
#include <rte_ring.h>
#include "qwctl.h"
#include "../include/conf.h"
/**
* help command
*/
struct cmd_help_tokens {
cmdline_fixed_string_t verb;
};
cmdline_parse_token_string_t cmd_help_verb =
TOKEN_STRING_INITIALIZER(struct cmd_help_tokens, verb, "help");
static void
cmd_help_handler(__attribute__((unused)) void *parsed_result,
struct cmdline *cl,
__attribute__((unused)) void *data)
{
cmdline_printf(cl, "Available commands:\n"
"- help\n"
"- set [ring_name|variable] <value>\n"
"- show [ring_name|variable]\n"
"\n"
"Available variables:\n"
"- low_watermark\n"
"- quota\n"
"- ring names follow the core%%u_port%%u format\n");
}
cmdline_parse_inst_t cmd_help = {
.f = cmd_help_handler,
.data = NULL,
.help_str = "show help",
.tokens = {
(void *) &cmd_help_verb,
NULL,
},
};
/**
* set command
*/
struct cmd_set_tokens {
cmdline_fixed_string_t verb;
cmdline_fixed_string_t variable;
uint32_t value;
};
cmdline_parse_token_string_t cmd_set_verb =
TOKEN_STRING_INITIALIZER(struct cmd_set_tokens, verb, "set");
cmdline_parse_token_string_t cmd_set_variable =
TOKEN_STRING_INITIALIZER(struct cmd_set_tokens, variable, NULL);
cmdline_parse_token_num_t cmd_set_value =
TOKEN_NUM_INITIALIZER(struct cmd_set_tokens, value, UINT32);
static void
cmd_set_handler(__attribute__((unused)) void *parsed_result,
struct cmdline *cl,
__attribute__((unused)) void *data)
{
struct cmd_set_tokens *tokens = parsed_result;
struct rte_ring *ring;
if (!strcmp(tokens->variable, "quota")) {
if (tokens->value > 0 && tokens->value <= MAX_PKT_QUOTA)
*quota = tokens->value;
else
cmdline_printf(cl, "quota must be between 1 and %u\n", MAX_PKT_QUOTA);
}
else if (!strcmp(tokens->variable, "low_watermark")) {
if (tokens->value <= 100)
*low_watermark = tokens->value * RING_SIZE / 100;
else
cmdline_printf(cl, "low_watermark must be between 0%% and 100%%\n");
}
else {
ring = rte_ring_lookup(tokens->variable);
if (ring == NULL)
cmdline_printf(cl, "Cannot find ring \"%s\"\n", tokens->variable);
else
if (tokens->value >= *low_watermark * 100 / RING_SIZE
&& tokens->value <= 100)
rte_ring_set_water_mark(ring, tokens->value * RING_SIZE / 100);
else
cmdline_printf(cl, "ring high watermark must be between %u%% "
"and 100%%\n", *low_watermark * 100 / RING_SIZE);
}
}
cmdline_parse_inst_t cmd_set = {
.f = cmd_set_handler,
.data = NULL,
.help_str = "Set a variable value",
.tokens = {
(void *) &cmd_set_verb,
(void *) &cmd_set_variable,
(void *) &cmd_set_value,
NULL,
},
};
/**
* show command
*/
struct cmd_show_tokens {
cmdline_fixed_string_t verb;
cmdline_fixed_string_t variable;
};
cmdline_parse_token_string_t cmd_show_verb =
TOKEN_STRING_INITIALIZER(struct cmd_show_tokens, verb, "show");
cmdline_parse_token_string_t cmd_show_variable =
TOKEN_STRING_INITIALIZER(struct cmd_show_tokens, variable, NULL);
static void
cmd_show_handler(__attribute__((unused)) void *parsed_result,
struct cmdline *cl,
__attribute__((unused)) void *data)
{
struct cmd_show_tokens *tokens = parsed_result;
struct rte_ring *ring;
if (!strcmp(tokens->variable, "quota"))
cmdline_printf(cl, "Global quota: %d\n", *quota);
else if (!strcmp(tokens->variable, "low_watermark"))
cmdline_printf(cl, "Global low_watermark: %u\n", *low_watermark);
else {
ring = rte_ring_lookup(tokens->variable);
if (ring == NULL)
cmdline_printf(cl, "Cannot find ring \"%s\"\n", tokens->variable);
else
rte_ring_dump(stdout, ring);
}
}
cmdline_parse_inst_t cmd_show = {
.f = cmd_show_handler,
.data = NULL,
.help_str = "Show a variable value",
.tokens = {
(void *) &cmd_show_verb,
(void *) &cmd_show_variable,
NULL,
},
};
cmdline_parse_ctx_t qwctl_ctx[] = {
(cmdline_parse_inst_t *)&cmd_help,
(cmdline_parse_inst_t *)&cmd_set,
(cmdline_parse_inst_t *)&cmd_show,
NULL,
};