Building with NEWLIBC triggers warnings about snprintf since stdio is no longer automatically included by printk.h This PR switches to using snprintk to avoid these warnings. Fixes #77330 Signed-off-by: Georges Oates_Larsen <georges.larsen@nordicsemi.no>
953 lines
22 KiB
C
953 lines
22 KiB
C
/*
|
|
* Copyright (c) 2024 Nordic Semiconductor ASA
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#include <zephyr/logging/log.h>
|
|
LOG_MODULE_DECLARE(net_shell);
|
|
|
|
#include <zephyr/net/net_if.h>
|
|
#include <strings.h>
|
|
#include <ctype.h>
|
|
|
|
#include "net_shell_private.h"
|
|
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
|
|
#include "conn_mgr_private.h"
|
|
#include <zephyr/net/conn_mgr_connectivity.h>
|
|
#include <zephyr/net/conn_mgr_monitor.h>
|
|
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
|
|
|
|
#define CM_IF_NAME_NONE "unnamed"
|
|
|
|
#if defined(CONFIG_NET_INTERFACE_NAME)
|
|
#define CM_MAX_IF_NAME MAX(sizeof(CM_IF_NAME_NONE), CONFIG_NET_INTERFACE_NAME_LEN)
|
|
#else
|
|
#define CM_MAX_IF_NAME sizeof(CM_IF_NAME_NONE)
|
|
#endif
|
|
|
|
#define CM_MAX_IF_INFO (CM_MAX_IF_NAME + 40)
|
|
|
|
|
|
/* Parsing and printing helpers. None of these are used unless CONFIG_NET_CONNECTION_MANAGER
|
|
* is enabled.
|
|
*/
|
|
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
|
|
enum cm_type {
|
|
CM_TARG_IFACE,
|
|
CM_TARG_NONE,
|
|
CM_TARG_ALL,
|
|
CM_TARG_INVALID,
|
|
};
|
|
struct cm_target {
|
|
enum cm_type type;
|
|
struct net_if *iface;
|
|
};
|
|
enum cm_gs_type {
|
|
CM_GS_GET,
|
|
CM_GS_SET
|
|
};
|
|
struct cm_flag_string {
|
|
const char *const name;
|
|
enum conn_mgr_if_flag flag;
|
|
};
|
|
static const struct cm_flag_string flag_strings[] = {
|
|
{"PERSISTENT", CONN_MGR_IF_PERSISTENT},
|
|
{"NO_AUTO_CONNECT", CONN_MGR_IF_NO_AUTO_CONNECT},
|
|
{"NO_AUTO_DOWN", CONN_MGR_IF_NO_AUTO_DOWN},
|
|
};
|
|
|
|
static const char *flag_name(enum conn_mgr_if_flag flag)
|
|
{
|
|
/* Scan over predefined flag strings, and return the name
|
|
* of the first one of matching flag.
|
|
*/
|
|
for (int i = 0; i < ARRAY_SIZE(flag_strings); i++) {
|
|
if (flag_strings[i].flag == flag) {
|
|
return flag_strings[i].name;
|
|
}
|
|
}
|
|
|
|
/* No matches found, it's invalid. */
|
|
return "INVALID";
|
|
}
|
|
|
|
static void cm_print_flags(const struct shell *sh)
|
|
{
|
|
PR("Valid flag keywords are:\n");
|
|
for (int i = 0; i < ARRAY_SIZE(flag_strings); i++) {
|
|
PR("\t%s,\n", flag_strings[i].name);
|
|
}
|
|
}
|
|
|
|
/* Verify that a provided string consists only of the characters 0-9*/
|
|
static bool check_numeric(char *str)
|
|
{
|
|
int i;
|
|
int len = strlen(str);
|
|
|
|
for (i = 0; i < len; i++) {
|
|
if (!isdigit((int)str[i])) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static void cm_target_help(const struct shell *sh)
|
|
{
|
|
PR("Valid target specifiers are 'ifi [index]', 'if [name]', or '[index]'.\n");
|
|
}
|
|
|
|
/* These parsers treat argv as a tokenstream, and increment *argidx by the number of
|
|
* tokens parsed.
|
|
*/
|
|
static int parse_ifi_target(const struct shell *sh, size_t argc, char *argv[], int *argidx,
|
|
struct cm_target *target)
|
|
{
|
|
char *arg;
|
|
int err = 0;
|
|
unsigned long iface_index;
|
|
|
|
/* At least one remaining argument is required to specify a target index */
|
|
if (*argidx >= argc) {
|
|
PR_ERROR("Please specify the target iface index.\n");
|
|
goto error;
|
|
}
|
|
|
|
arg = argv[*argidx];
|
|
|
|
iface_index = shell_strtoul(arg, 10, &err);
|
|
|
|
if (err) {
|
|
PR_ERROR("\"%s\" is not a valid iface index.\n", arg);
|
|
goto error;
|
|
}
|
|
|
|
target->iface = net_if_get_by_index(iface_index);
|
|
|
|
if (target->iface == NULL) {
|
|
PR_ERROR("iface with index \"%s\" does not exist.\n", arg);
|
|
goto error;
|
|
}
|
|
|
|
*argidx += 1;
|
|
target->type = CM_TARG_IFACE;
|
|
return 0;
|
|
|
|
error:
|
|
target->type = CM_TARG_INVALID;
|
|
return -EINVAL;
|
|
}
|
|
|
|
static int parse_if_target(const struct shell *sh, size_t argc, char *argv[], int *argidx,
|
|
struct cm_target *target)
|
|
{
|
|
#if defined(CONFIG_NET_INTERFACE_NAME)
|
|
char *arg;
|
|
/* At least one remaining argument is required to specify a target name */
|
|
if (*argidx >= argc) {
|
|
PR_ERROR("Please specify the target iface name.\n");
|
|
goto error;
|
|
}
|
|
|
|
arg = argv[*argidx];
|
|
|
|
target->iface = net_if_get_by_index(net_if_get_by_name(arg));
|
|
|
|
if (target->iface == NULL) {
|
|
PR_ERROR("iface with name \"%s\" does not exist.\n", arg);
|
|
goto error;
|
|
}
|
|
|
|
*argidx += 1;
|
|
target->type = CM_TARG_IFACE;
|
|
return 0;
|
|
#else
|
|
PR_ERROR("iface name lookup requires CONFIG_NET_INTERFACE_NAME.\n");
|
|
goto error;
|
|
#endif
|
|
|
|
error:
|
|
target->type = CM_TARG_INVALID;
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* parse `if [iface name]`, `ifi [iface index]`, `[iface index]`, or `all` */
|
|
static int parse_target(const struct shell *sh, size_t argc, char *argv[], int *argidx,
|
|
struct cm_target *target)
|
|
{
|
|
char *arg;
|
|
|
|
/* At least one argument is required to specify a target */
|
|
if (*argidx >= argc) {
|
|
target->type = CM_TARG_NONE;
|
|
return 0;
|
|
}
|
|
|
|
arg = argv[*argidx];
|
|
|
|
/* At least one argument provided. Is it "all" or "none"? */
|
|
if (strcasecmp(arg, "all") == 0) {
|
|
*argidx += 1;
|
|
target->type = CM_TARG_ALL;
|
|
return 0;
|
|
}
|
|
|
|
if (strcasecmp(arg, "none") == 0) {
|
|
*argidx += 1;
|
|
target->type = CM_TARG_NONE;
|
|
return 0;
|
|
}
|
|
|
|
/* If not, interpret as an iface index if it is also numeric */
|
|
if (check_numeric(arg)) {
|
|
return parse_ifi_target(sh, argc, argv, argidx, target);
|
|
}
|
|
|
|
/* Otherwise, arg must be a target type specifier */
|
|
if (strcasecmp(arg, "if") == 0) {
|
|
*argidx += 1;
|
|
return parse_if_target(sh, argc, argv, argidx, target);
|
|
}
|
|
|
|
if (strcasecmp(arg, "ifi") == 0) {
|
|
*argidx += 1;
|
|
return parse_ifi_target(sh, argc, argv, argidx, target);
|
|
}
|
|
|
|
PR_ERROR("%s is not a valid target type or target specifier.\n", arg);
|
|
cm_target_help(sh);
|
|
target->type = CM_TARG_INVALID;
|
|
|
|
return -EINVAL;
|
|
}
|
|
|
|
static int parse_getset(const struct shell *sh, size_t argc, char *argv[], int *argidx,
|
|
enum cm_gs_type *result)
|
|
{
|
|
char *arg;
|
|
|
|
/* At least one argument is required to specify get or set */
|
|
if (*argidx >= argc) {
|
|
goto error;
|
|
}
|
|
|
|
arg = argv[*argidx];
|
|
|
|
if (strcasecmp(arg, "get") == 0) {
|
|
*argidx += 1;
|
|
*result = CM_GS_GET;
|
|
return 0;
|
|
}
|
|
|
|
if (strcasecmp(arg, "set") == 0) {
|
|
*argidx += 1;
|
|
*result = CM_GS_SET;
|
|
return 0;
|
|
}
|
|
|
|
error:
|
|
PR_ERROR("Please specify get or set\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
static int parse_flag(const struct shell *sh, size_t argc, char *argv[], int *argidx,
|
|
enum conn_mgr_if_flag *result)
|
|
{
|
|
char *arg;
|
|
|
|
/* At least one argument is required to specify get or set */
|
|
if (*argidx >= argc) {
|
|
PR_ERROR("Please specify a flag.\n");
|
|
cm_print_flags(sh);
|
|
|
|
return -EINVAL;
|
|
}
|
|
|
|
arg = argv[*argidx];
|
|
|
|
for (int i = 0; i < ARRAY_SIZE(flag_strings); i++) {
|
|
if (strcasecmp(arg, flag_strings[i].name) == 0) {
|
|
*argidx += 1;
|
|
*result = flag_strings[i].flag;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
PR_ERROR("%s is not a valid flag.\n", arg);
|
|
return -EINVAL;
|
|
}
|
|
|
|
static int parse_bool(const struct shell *sh, size_t argc, char *argv[], int *argidx, bool *result)
|
|
{
|
|
char *arg;
|
|
|
|
/* At least one argument is required to specify a boolean */
|
|
if (*argidx >= argc) {
|
|
goto error;
|
|
}
|
|
|
|
arg = argv[*argidx];
|
|
|
|
if (strcasecmp(arg, "yes") == 0 ||
|
|
strcasecmp(arg, "y") == 0 ||
|
|
strcasecmp(arg, "1") == 0 ||
|
|
strcasecmp(arg, "true") == 0) {
|
|
*argidx += 1;
|
|
*result = true;
|
|
return 0;
|
|
}
|
|
|
|
if (strcasecmp(arg, "no") == 0 ||
|
|
strcasecmp(arg, "n") == 0 ||
|
|
strcasecmp(arg, "0") == 0 ||
|
|
strcasecmp(arg, "false") == 0) {
|
|
*argidx += 1;
|
|
*result = false;
|
|
return 0;
|
|
}
|
|
|
|
error:
|
|
PR_ERROR("Please specify true or false.\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
static int parse_timeout(const struct shell *sh, size_t argc, char *argv[], int *argidx,
|
|
int *result)
|
|
{
|
|
char *arg;
|
|
int err = 0;
|
|
unsigned long value;
|
|
|
|
/* At least one argument is required to specify a timeout */
|
|
if (*argidx >= argc) {
|
|
PR_ERROR("Please specify a timeout (in seconds).\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
arg = argv[*argidx];
|
|
|
|
/* Check for special keyword "none" */
|
|
if (strcasecmp(arg, "none") == 0) {
|
|
*argidx += 1;
|
|
*result = CONN_MGR_IF_NO_TIMEOUT;
|
|
return 0;
|
|
}
|
|
|
|
/* Otherwise, try to parse integer timeout (seconds). */
|
|
if (!check_numeric(arg)) {
|
|
PR_ERROR("%s is not a valid timeout.\n", arg);
|
|
return -EINVAL;
|
|
}
|
|
|
|
value = shell_strtoul(arg, 10, &err);
|
|
if (err) {
|
|
PR_ERROR("%s is not a valid timeout.\n", arg);
|
|
return -EINVAL;
|
|
}
|
|
|
|
*argidx += 1;
|
|
*result = value;
|
|
return 0;
|
|
}
|
|
|
|
static void cm_get_iface_info(struct net_if *iface, char *buf, size_t len)
|
|
{
|
|
#if defined(CONFIG_NET_INTERFACE_NAME)
|
|
char name[CM_MAX_IF_NAME];
|
|
|
|
if (net_if_get_name(iface, name, sizeof(name))) {
|
|
strcpy(name, CM_IF_NAME_NONE);
|
|
}
|
|
|
|
snprintk(buf, len, "%d (%p - %s - %s)", net_if_get_by_iface(iface), iface, name,
|
|
iface2str(iface, NULL));
|
|
#else
|
|
snprintk(buf, len, "%d (%p - %s)", net_if_get_by_iface(iface), iface,
|
|
iface2str(iface, NULL));
|
|
#endif
|
|
}
|
|
|
|
/* bulk iface actions */
|
|
static void cm_iface_status(struct net_if *iface, void *user_data)
|
|
{
|
|
const struct shell *sh = user_data;
|
|
uint16_t state = conn_mgr_if_state(iface);
|
|
bool ignored;
|
|
bool bound;
|
|
bool admin_up;
|
|
bool oper_up;
|
|
bool has_ipv4;
|
|
bool has_ipv6;
|
|
bool connected;
|
|
char iface_info[CM_MAX_IF_INFO];
|
|
char *ip_state;
|
|
|
|
cm_get_iface_info(iface, iface_info, sizeof(iface_info));
|
|
|
|
if (state == CONN_MGR_IF_STATE_INVALID) {
|
|
PR("iface %s not tracked.\n", iface_info);
|
|
} else {
|
|
ignored = state & CONN_MGR_IF_IGNORED;
|
|
bound = conn_mgr_if_is_bound(iface);
|
|
admin_up = net_if_is_admin_up(iface);
|
|
oper_up = state & CONN_MGR_IF_UP;
|
|
has_ipv4 = state & CONN_MGR_IF_IPV4_SET;
|
|
has_ipv6 = state & CONN_MGR_IF_IPV6_SET;
|
|
connected = state & CONN_MGR_IF_READY;
|
|
|
|
if (has_ipv4 && has_ipv6) {
|
|
ip_state = "IPv4 + IPv6";
|
|
} else if (has_ipv4) {
|
|
ip_state = "IPv4";
|
|
} else if (has_ipv6) {
|
|
ip_state = "IPv6";
|
|
} else {
|
|
ip_state = "no IP";
|
|
}
|
|
|
|
PR("iface %s status: %s, %s, %s, %s, %s, %s.\n", iface_info,
|
|
ignored ? "ignored" : "watched",
|
|
bound ? "bound" : "not bound",
|
|
admin_up ? "admin-up" : "admin-down",
|
|
oper_up ? "oper-up" : "oper-down",
|
|
ip_state,
|
|
connected ? "connected" : "not connected");
|
|
}
|
|
}
|
|
|
|
static void cm_iface_ignore(struct net_if *iface, void *user_data)
|
|
{
|
|
const struct shell *sh = user_data;
|
|
char iface_info[CM_MAX_IF_INFO];
|
|
|
|
cm_get_iface_info(iface, iface_info, sizeof(iface_info));
|
|
|
|
conn_mgr_ignore_iface(iface);
|
|
PR("iface %s now ignored.\n", iface_info);
|
|
}
|
|
|
|
static void cm_iface_watch(struct net_if *iface, void *user_data)
|
|
{
|
|
const struct shell *sh = user_data;
|
|
char iface_info[CM_MAX_IF_INFO];
|
|
|
|
cm_get_iface_info(iface, iface_info, sizeof(iface_info));
|
|
|
|
conn_mgr_watch_iface(iface);
|
|
PR("iface %s now watched.\n", iface_info);
|
|
}
|
|
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
|
|
static void not_available(const struct shell *sh)
|
|
{
|
|
PR_INFO("This command is not available unless CONFIG_NET_CONNECTION_MANAGER is enabled.\n");
|
|
}
|
|
|
|
#endif /* !defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
|
|
|
|
/* Commands */
|
|
|
|
static int cmd_net_cm_status(const struct shell *sh, size_t argc, char *argv[])
|
|
{
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
int argidx = 1;
|
|
struct cm_target target = {
|
|
.type = CM_TARG_INVALID
|
|
};
|
|
|
|
if (parse_target(sh, argc, argv, &argidx, &target)) {
|
|
/* no need to print anything, parse_target already explained the issue */
|
|
return 0;
|
|
}
|
|
|
|
if (argidx != argc) {
|
|
PR_ERROR("Too many args.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_NONE || target.type == CM_TARG_ALL) {
|
|
net_if_foreach(cm_iface_status, (void *)sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_IFACE) {
|
|
cm_iface_status(target.iface, (void *)sh);
|
|
return 0;
|
|
}
|
|
|
|
PR_ERROR("Invalid target selected.\n");
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
PR_INFO("conn_mgr is not enabled. Enable by setting CONFIG_NET_CONNECTION_MANAGER=y.\n");
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_net_cm_ignore(const struct shell *sh, size_t argc, char *argv[])
|
|
{
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
int argidx = 1;
|
|
struct cm_target target = {
|
|
.type = CM_TARG_INVALID
|
|
};
|
|
|
|
if (parse_target(sh, argc, argv, &argidx, &target)) {
|
|
/* no need to print anything, parse_target already explained the issue */
|
|
return 0;
|
|
}
|
|
|
|
if (argidx != argc) {
|
|
PR_ERROR("Too many args.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_NONE) {
|
|
PR_ERROR("Please specify a target.\n");
|
|
cm_target_help(sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_ALL) {
|
|
PR("Ignoring all ifaces.\n");
|
|
net_if_foreach(cm_iface_ignore, (void *)sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_IFACE) {
|
|
cm_iface_ignore(target.iface, (void *)sh);
|
|
return 0;
|
|
}
|
|
|
|
PR_ERROR("Invalid target selected.\n");
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
not_available(sh);
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_net_cm_watch(const struct shell *sh, size_t argc, char *argv[])
|
|
{
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
int argidx = 1;
|
|
struct cm_target target = {
|
|
.type = CM_TARG_INVALID
|
|
};
|
|
|
|
if (parse_target(sh, argc, argv, &argidx, &target)) {
|
|
/* no need to print anything, parse_target already explained the issue */
|
|
return 0;
|
|
}
|
|
|
|
if (argidx != argc) {
|
|
PR_ERROR("Too many args.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_NONE) {
|
|
PR_ERROR("Please specify a target.\n");
|
|
cm_target_help(sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_ALL) {
|
|
PR("Watching all ifaces.\n");
|
|
net_if_foreach(cm_iface_watch, (void *)sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_IFACE) {
|
|
cm_iface_watch(target.iface, (void *)sh);
|
|
return 0;
|
|
}
|
|
|
|
PR_ERROR("Invalid target selected.\n");
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
not_available(sh);
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_net_cm_connect(const struct shell *sh, size_t argc, char *argv[])
|
|
{
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
int argidx = 1;
|
|
struct cm_target target = {
|
|
.type = CM_TARG_INVALID
|
|
};
|
|
char iface_info[CM_MAX_IF_INFO];
|
|
|
|
if (parse_target(sh, argc, argv, &argidx, &target)) {
|
|
/* no need to print anything, parse_target already explained the issue */
|
|
return 0;
|
|
}
|
|
|
|
if (argidx != argc) {
|
|
PR_ERROR("Too many args.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_NONE) {
|
|
PR_ERROR("Please specify a target.\n");
|
|
cm_target_help(sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_ALL) {
|
|
PR("Instructing all non-ignored ifaces to connect.\n");
|
|
conn_mgr_all_if_connect(true);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_IFACE) {
|
|
cm_get_iface_info(target.iface, iface_info, sizeof(iface_info));
|
|
|
|
if (!conn_mgr_if_is_bound(target.iface)) {
|
|
PR_ERROR("iface %s is not bound to a connectivity implementation, cannot "
|
|
"connect.\n", iface_info);
|
|
return 0;
|
|
}
|
|
|
|
PR("Instructing iface %s to connect.\n", iface_info);
|
|
conn_mgr_if_connect(target.iface);
|
|
return 0;
|
|
}
|
|
|
|
PR_ERROR("Invalid target selected.\n");
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
not_available(sh);
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_net_cm_disconnect(const struct shell *sh, size_t argc, char *argv[])
|
|
{
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
int argidx = 1;
|
|
struct cm_target target = {
|
|
.type = CM_TARG_INVALID
|
|
};
|
|
char iface_info[CM_MAX_IF_INFO];
|
|
|
|
if (parse_target(sh, argc, argv, &argidx, &target)) {
|
|
/* no need to print anything, parse_target already explained the issue */
|
|
return 0;
|
|
}
|
|
|
|
if (argidx != argc) {
|
|
PR_ERROR("Too many args.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_NONE) {
|
|
PR_ERROR("Please specify a target.\n");
|
|
cm_target_help(sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_ALL) {
|
|
PR("Instructing all non-ignored ifaces to disconnect.\n");
|
|
conn_mgr_all_if_disconnect(true);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_IFACE) {
|
|
cm_get_iface_info(target.iface, iface_info, sizeof(iface_info));
|
|
|
|
if (!conn_mgr_if_is_bound(target.iface)) {
|
|
PR_ERROR("iface %s is not bound to a connectivity implementation, cannot "
|
|
"disconnect.\n", iface_info);
|
|
return 0;
|
|
}
|
|
|
|
PR("Instructing iface %s to disonnect.\n", iface_info);
|
|
conn_mgr_if_disconnect(target.iface);
|
|
return 0;
|
|
}
|
|
|
|
PR_ERROR("Invalid target selected.\n");
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
not_available(sh);
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_net_cm_up(const struct shell *sh, size_t argc, char *argv[])
|
|
{
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
int argidx = 1;
|
|
struct cm_target target = {
|
|
.type = CM_TARG_INVALID
|
|
};
|
|
char iface_info[CM_MAX_IF_INFO];
|
|
|
|
if (parse_target(sh, argc, argv, &argidx, &target)) {
|
|
/* no need to print anything, parse_target already explained the issue */
|
|
return 0;
|
|
}
|
|
|
|
if (argidx != argc) {
|
|
PR_ERROR("Too many args.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_NONE) {
|
|
PR_ERROR("Please specify a target.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_ALL) {
|
|
PR("Taking all non-ignored ifaces admin-up.\n");
|
|
conn_mgr_all_if_up(true);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_IFACE) {
|
|
cm_get_iface_info(target.iface, iface_info, sizeof(iface_info));
|
|
PR("Taking iface %s admin-up.\n", iface_info);
|
|
PR_WARNING("This command duplicates 'net iface up' if [target] != all.\n");
|
|
|
|
net_if_up(target.iface);
|
|
return 0;
|
|
}
|
|
|
|
PR_ERROR("Invalid target selected.\n");
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
not_available(sh);
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_net_cm_down(const struct shell *sh, size_t argc, char *argv[])
|
|
{
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
int argidx = 1;
|
|
struct cm_target target = {
|
|
.type = CM_TARG_INVALID
|
|
};
|
|
char iface_info[CM_MAX_IF_INFO];
|
|
|
|
if (parse_target(sh, argc, argv, &argidx, &target)) {
|
|
/* no need to print anything, parse_target already explained the issue */
|
|
return 0;
|
|
}
|
|
|
|
if (argidx != argc) {
|
|
PR_ERROR("Too many args.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_NONE) {
|
|
PR_ERROR("Please specify a target.\n");
|
|
cm_target_help(sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_ALL) {
|
|
PR("Taking all non-ignored ifaces admin-down.\n");
|
|
conn_mgr_all_if_down(true);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_IFACE) {
|
|
cm_get_iface_info(target.iface, iface_info, sizeof(iface_info));
|
|
PR("Taking iface %s admin-down.\n", iface_info);
|
|
PR_WARNING("This command duplicates 'net iface down' if [target] != all.\n");
|
|
|
|
net_if_down(target.iface);
|
|
return 0;
|
|
}
|
|
|
|
PR_ERROR("Invalid target selected.\n");
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
not_available(sh);
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_net_cm_flag(const struct shell *sh, size_t argc, char *argv[])
|
|
{
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
int argidx = 1;
|
|
enum cm_gs_type getset = CM_GS_GET;
|
|
enum conn_mgr_if_flag flag = CONN_MGR_IF_PERSISTENT;
|
|
bool value = false;
|
|
struct cm_target target = {
|
|
.type = CM_TARG_INVALID
|
|
};
|
|
char iface_info[CM_MAX_IF_INFO];
|
|
|
|
if (parse_target(sh, argc, argv, &argidx, &target)) {
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_NONE) {
|
|
PR_ERROR("Please specify a target.\n");
|
|
cm_target_help(sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_ALL) {
|
|
PR_ERROR("Cannot get/set flags for all ifaces.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type != CM_TARG_IFACE) {
|
|
PR_ERROR("Invalid target selected.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (parse_getset(sh, argc, argv, &argidx, &getset)) {
|
|
return 0;
|
|
}
|
|
|
|
if (parse_flag(sh, argc, argv, &argidx, &flag)) {
|
|
return 0;
|
|
}
|
|
|
|
/* If we are in set mode, expect the value to be provided. */
|
|
if (getset == CM_GS_SET && parse_bool(sh, argc, argv, &argidx, &value)) {
|
|
return 0;
|
|
}
|
|
|
|
if (argidx != argc) {
|
|
PR_ERROR("Too many args.\n");
|
|
return 0;
|
|
}
|
|
|
|
cm_get_iface_info(target.iface, iface_info, sizeof(iface_info));
|
|
|
|
if (!conn_mgr_if_is_bound(target.iface)) {
|
|
PR_ERROR("iface %s is not bound to a connectivity implementation, cannot "
|
|
"get/set connectivity flag.\n", iface_info);
|
|
return 0;
|
|
}
|
|
|
|
if (getset == CM_GS_SET) {
|
|
(void)conn_mgr_if_set_flag(target.iface, flag, value);
|
|
PR("Set the connectivity %s flag to %s on iface %s.\n", flag_name(flag),
|
|
value?"y":"n", iface_info);
|
|
} else {
|
|
value = conn_mgr_if_get_flag(target.iface, flag);
|
|
PR("The current value of the %s connectivity flag on iface %s is %s.\n",
|
|
flag_name(flag), iface_info, value?"y":"n");
|
|
}
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
not_available(sh);
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_net_cm_timeout(const struct shell *sh, size_t argc, char *argv[])
|
|
{
|
|
#if defined(CONFIG_NET_CONNECTION_MANAGER)
|
|
int argidx = 1;
|
|
enum cm_gs_type getset = CM_GS_GET;
|
|
int value = CONN_MGR_IF_NO_TIMEOUT;
|
|
struct cm_target target = {
|
|
.type = CM_TARG_INVALID
|
|
};
|
|
char iface_info[CM_MAX_IF_INFO];
|
|
|
|
if (parse_target(sh, argc, argv, &argidx, &target)) {
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_NONE) {
|
|
PR_ERROR("Please specify a target.\n");
|
|
cm_target_help(sh);
|
|
return 0;
|
|
}
|
|
|
|
if (target.type == CM_TARG_ALL) {
|
|
PR_ERROR("Cannot get/set timeout for all ifaces.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (target.type != CM_TARG_IFACE) {
|
|
PR_ERROR("Invalid target selected.\n");
|
|
return 0;
|
|
}
|
|
|
|
if (parse_getset(sh, argc, argv, &argidx, &getset)) {
|
|
return 0;
|
|
}
|
|
|
|
/* If we are in set mode, expect the value to be provided. */
|
|
if (getset == CM_GS_SET && parse_timeout(sh, argc, argv, &argidx, &value)) {
|
|
return 0;
|
|
}
|
|
|
|
if (argidx != argc) {
|
|
PR_ERROR("Too many args.\n");
|
|
return 0;
|
|
}
|
|
|
|
cm_get_iface_info(target.iface, iface_info, sizeof(iface_info));
|
|
|
|
if (!conn_mgr_if_is_bound(target.iface)) {
|
|
PR_ERROR("iface %s is not bound to a connectivity implementation, cannot "
|
|
"get/set connectivity timeout.\n", iface_info);
|
|
return 0;
|
|
}
|
|
|
|
if (getset == CM_GS_SET) {
|
|
(void)conn_mgr_if_set_timeout(target.iface, value);
|
|
PR("Set the connectivity timeout for iface %s to %d%s.\n", iface_info, value,
|
|
value == 0 ? " (no timeout)":" seconds");
|
|
} else {
|
|
value = conn_mgr_if_get_timeout(target.iface);
|
|
PR("The connectivity timeout for iface %s is %d%s.\n", iface_info, value,
|
|
value == 0 ? " (no timeout)":" seconds");
|
|
}
|
|
#else /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
not_available(sh);
|
|
#endif /* defined(CONFIG_NET_CONNECTION_MANAGER) */
|
|
return 0;
|
|
}
|
|
|
|
SHELL_STATIC_SUBCMD_SET_CREATE(net_cmd_cm,
|
|
SHELL_CMD_ARG(status, NULL,
|
|
"'net cm status [target]' shows the connectivity status of the specified "
|
|
"iface(s).",
|
|
cmd_net_cm_status, 1, 2),
|
|
SHELL_CMD_ARG(ignore, NULL,
|
|
"'net cm ignore [target]' ignores the specified iface(s).",
|
|
cmd_net_cm_ignore, 1, 2),
|
|
SHELL_CMD_ARG(watch, NULL,
|
|
"'net cm watch [target]' watches the specified iface(s).",
|
|
cmd_net_cm_watch, 1, 2),
|
|
SHELL_CMD_ARG(connect, NULL,
|
|
"'net cm connect [target]' connects the specified iface(s).",
|
|
cmd_net_cm_connect, 1, 2),
|
|
SHELL_CMD_ARG(disconnect, NULL,
|
|
"'net cm disconnect [target]' disconnects the specified iface(s).",
|
|
cmd_net_cm_disconnect, 1, 2),
|
|
SHELL_CMD_ARG(up, NULL,
|
|
"'net cm up [target]' takes the specified iface(s) admin-up.",
|
|
cmd_net_cm_up, 1, 2),
|
|
SHELL_CMD_ARG(down, NULL,
|
|
"'net cm down [target]' takes the specified iface(s) admin-down.",
|
|
cmd_net_cm_down, 1, 2),
|
|
SHELL_CMD_ARG(flag, NULL,
|
|
"'net cm flag [target] [get/set] [flag] [value]' gets or sets a flag "
|
|
"for the specified iface.",
|
|
cmd_net_cm_flag, 1, 5),
|
|
SHELL_CMD_ARG(timeout, NULL,
|
|
"'net cm timeout [target] [get/set] [value]' gets or sets the timeout "
|
|
"for the specified iface.",
|
|
cmd_net_cm_timeout, 1, 4),
|
|
SHELL_SUBCMD_SET_END
|
|
);
|
|
|
|
SHELL_SUBCMD_ADD((net), cm, &net_cmd_cm, "Control conn_mgr.", NULL, 1, 0);
|