123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347 |
- /*
- * Copyright 2015 Freescale Semiconductor, Inc.
- *
- * SPDX-License-Identifier: GPL-2.0+
- *
- * Ethernet Switch commands
- */
- #include <common.h>
- #include <command.h>
- #include <errno.h>
- #include <ethsw.h>
- static const char *ethsw_name;
- static struct keywords_to_function {
- enum ethsw_keyword_id cmd_keyword[ETHSW_MAX_CMD_PARAMS];
- int cmd_func_offset;
- int (*keyword_function)(struct ethsw_command_def *parsed_cmd);
- } ethsw_cmd_def[] = {
- {
- .cmd_keyword = {
- ethsw_id_enable,
- ethsw_id_key_end,
- },
- .cmd_func_offset = offsetof(struct ethsw_command_func,
- port_enable),
- .keyword_function = NULL,
- }, {
- .cmd_keyword = {
- ethsw_id_disable,
- ethsw_id_key_end,
- },
- .cmd_func_offset = offsetof(struct ethsw_command_func,
- port_disable),
- .keyword_function = NULL,
- }, {
- .cmd_keyword = {
- ethsw_id_show,
- ethsw_id_key_end,
- },
- .cmd_func_offset = offsetof(struct ethsw_command_func,
- port_show),
- .keyword_function = NULL,
- },
- };
- struct keywords_optional {
- int cmd_keyword[ETHSW_MAX_CMD_PARAMS];
- } cmd_opt_def[] = {
- {
- .cmd_keyword = {
- ethsw_id_port,
- ethsw_id_port_no,
- ethsw_id_key_end,
- },
- },
- };
- static int keyword_match_gen(enum ethsw_keyword_id key_id, int argc, char
- *const argv[], int *argc_nr,
- struct ethsw_command_def *parsed_cmd);
- static int keyword_match_port(enum ethsw_keyword_id key_id, int argc,
- char *const argv[], int *argc_nr,
- struct ethsw_command_def *parsed_cmd);
- /*
- * Define properties for each keyword;
- * keep the order synced with enum ethsw_keyword_id
- */
- struct keyword_def {
- const char *keyword_name;
- int (*match)(enum ethsw_keyword_id key_id, int argc, char *const argv[],
- int *argc_nr, struct ethsw_command_def *parsed_cmd);
- } keyword[] = {
- {
- .keyword_name = "help",
- .match = &keyword_match_gen,
- }, {
- .keyword_name = "show",
- .match = &keyword_match_gen,
- }, {
- .keyword_name = "port",
- .match = &keyword_match_port
- }, {
- .keyword_name = "enable",
- .match = &keyword_match_gen,
- }, {
- .keyword_name = "disable",
- .match = &keyword_match_gen,
- },
- };
- /*
- * Function used by an Ethernet Switch driver to set the functions
- * that must be called by the parser when an ethsw command is given
- */
- int ethsw_define_functions(const struct ethsw_command_func *cmd_func)
- {
- int i;
- void **aux_p;
- int (*cmd_func_aux)(struct ethsw_command_def *);
- if (!cmd_func->ethsw_name)
- return -EINVAL;
- ethsw_name = cmd_func->ethsw_name;
- for (i = 0; i < ARRAY_SIZE(ethsw_cmd_def); i++) {
- /*
- * get the pointer to the function send by the Ethernet Switch
- * driver that corresponds to the proper ethsw command
- */
- if (ethsw_cmd_def[i].keyword_function)
- continue;
- aux_p = (void *)cmd_func + ethsw_cmd_def[i].cmd_func_offset;
- cmd_func_aux = (int (*)(struct ethsw_command_def *)) *aux_p;
- ethsw_cmd_def[i].keyword_function = cmd_func_aux;
- }
- return 0;
- }
- /* Generic function used to match a keyword only by a string */
- static int keyword_match_gen(enum ethsw_keyword_id key_id, int argc,
- char *const argv[], int *argc_nr,
- struct ethsw_command_def *parsed_cmd)
- {
- if (strcmp(argv[*argc_nr], keyword[key_id].keyword_name) == 0) {
- parsed_cmd->cmd_to_keywords[*argc_nr] = key_id;
- return 1;
- }
- return 0;
- }
- /* Function used to match the command's port */
- static int keyword_match_port(enum ethsw_keyword_id key_id, int argc,
- char *const argv[], int *argc_nr,
- struct ethsw_command_def *parsed_cmd)
- {
- unsigned long val;
- if (!keyword_match_gen(key_id, argc, argv, argc_nr, parsed_cmd))
- return 0;
- if (*argc_nr + 1 >= argc)
- return 0;
- if (strict_strtoul(argv[*argc_nr + 1], 10, &val) != -EINVAL) {
- parsed_cmd->port = val;
- (*argc_nr)++;
- parsed_cmd->cmd_to_keywords[*argc_nr] = ethsw_id_port_no;
- return 1;
- }
- return 0;
- }
- /* Finds optional keywords and modifies *argc_va to skip them */
- static void cmd_keywords_opt_check(const struct ethsw_command_def *parsed_cmd,
- int *argc_val)
- {
- int i;
- int keyw_opt_matched;
- int argc_val_max;
- int const *cmd_keyw_p;
- int const *cmd_keyw_opt_p;
- /* remember the best match */
- argc_val_max = *argc_val;
- /*
- * check if our command's optional keywords match the optional
- * keywords of an available command
- */
- for (i = 0; i < ARRAY_SIZE(ethsw_cmd_def); i++) {
- keyw_opt_matched = 0;
- cmd_keyw_p = &parsed_cmd->cmd_to_keywords[keyw_opt_matched];
- cmd_keyw_opt_p = &cmd_opt_def[i].cmd_keyword[keyw_opt_matched];
- /*
- * increase the number of keywords that
- * matched with a command
- */
- while (keyw_opt_matched + *argc_val <
- parsed_cmd->cmd_keywords_nr &&
- *cmd_keyw_opt_p != ethsw_id_key_end &&
- *(cmd_keyw_p + *argc_val) == *cmd_keyw_opt_p) {
- keyw_opt_matched++;
- cmd_keyw_p++;
- cmd_keyw_opt_p++;
- }
- /*
- * if all our optional command's keywords perfectly match an
- * optional pattern, then we can move to the next defined
- * keywords in our command; remember the one that matched the
- * greatest number of keywords
- */
- if (keyw_opt_matched + *argc_val <=
- parsed_cmd->cmd_keywords_nr &&
- *cmd_keyw_opt_p == ethsw_id_key_end &&
- *argc_val + keyw_opt_matched > argc_val_max)
- argc_val_max = *argc_val + keyw_opt_matched;
- }
- *argc_val = argc_val_max;
- }
- /*
- * Finds the function to call based on keywords and
- * modifies *argc_va to skip them
- */
- static void cmd_keywords_check(struct ethsw_command_def *parsed_cmd,
- int *argc_val)
- {
- int i;
- int keyw_matched;
- int *cmd_keyw_p;
- int *cmd_keyw_def_p;
- /*
- * check if our command's keywords match the
- * keywords of an available command
- */
- for (i = 0; i < ARRAY_SIZE(ethsw_cmd_def); i++) {
- keyw_matched = 0;
- cmd_keyw_p = &parsed_cmd->cmd_to_keywords[keyw_matched];
- cmd_keyw_def_p = ðsw_cmd_def[i].cmd_keyword[keyw_matched];
- /*
- * increase the number of keywords that
- * matched with a command
- */
- while (keyw_matched + *argc_val < parsed_cmd->cmd_keywords_nr &&
- *cmd_keyw_def_p != ethsw_id_key_end &&
- *(cmd_keyw_p + *argc_val) == *cmd_keyw_def_p) {
- keyw_matched++;
- cmd_keyw_p++;
- cmd_keyw_def_p++;
- }
- /*
- * if all our command's keywords perfectly match an
- * available command, then we get the function we need to call
- * to configure the Ethernet Switch
- */
- if (keyw_matched && keyw_matched + *argc_val ==
- parsed_cmd->cmd_keywords_nr &&
- *cmd_keyw_def_p == ethsw_id_key_end) {
- *argc_val += keyw_matched;
- parsed_cmd->cmd_function =
- ethsw_cmd_def[i].keyword_function;
- return;
- }
- }
- }
- /* find all the keywords in the command */
- static int keywords_find(int argc, char * const argv[],
- struct ethsw_command_def *parsed_cmd)
- {
- int i;
- int j;
- int argc_val;
- int rc = CMD_RET_SUCCESS;
- for (i = 1; i < argc; i++) {
- for (j = 0; j < ethsw_id_count; j++) {
- if (keyword[j].match(j, argc, argv, &i, parsed_cmd))
- break;
- }
- }
- /* if there is no keyword match for a word, the command is invalid */
- for (i = 1; i < argc; i++)
- if (parsed_cmd->cmd_to_keywords[i] == ethsw_id_key_end)
- rc = CMD_RET_USAGE;
- parsed_cmd->cmd_keywords_nr = argc;
- argc_val = 1;
- /* get optional parameters first */
- cmd_keywords_opt_check(parsed_cmd, &argc_val);
- if (argc_val == parsed_cmd->cmd_keywords_nr)
- return CMD_RET_USAGE;
- /*
- * check the keywords and if a match is found,
- * get the function to call
- */
- cmd_keywords_check(parsed_cmd, &argc_val);
- /* error if not all commands' parameters were matched */
- if (argc_val == parsed_cmd->cmd_keywords_nr) {
- if (!parsed_cmd->cmd_function) {
- printf("Command not available for: %s\n", ethsw_name);
- rc = CMD_RET_FAILURE;
- }
- } else {
- rc = CMD_RET_USAGE;
- }
- return rc;
- }
- static void command_def_init(struct ethsw_command_def *parsed_cmd)
- {
- int i;
- for (i = 0; i < ETHSW_MAX_CMD_PARAMS; i++)
- parsed_cmd->cmd_to_keywords[i] = ethsw_id_key_end;
- parsed_cmd->port = ETHSW_CMD_PORT_ALL;
- parsed_cmd->cmd_function = NULL;
- }
- /* function to interpret commands starting with "ethsw " */
- static int do_ethsw(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
- {
- struct ethsw_command_def parsed_cmd;
- int rc = CMD_RET_SUCCESS;
- if (argc == 1 || argc >= ETHSW_MAX_CMD_PARAMS)
- return CMD_RET_USAGE;
- command_def_init(&parsed_cmd);
- rc = keywords_find(argc, argv, &parsed_cmd);
- if (rc == CMD_RET_SUCCESS)
- rc = parsed_cmd.cmd_function(&parsed_cmd);
- return rc;
- }
- #define ETHSW_PORT_CONF_HELP "[port <port_no>] { enable | disable | show } " \
- "- enable/disable a port; show shows a port's configuration"
- U_BOOT_CMD(ethsw, ETHSW_MAX_CMD_PARAMS, 0, do_ethsw,
- "Ethernet l2 switch commands",
- ETHSW_PORT_CONF_HELP"\n"
- );
|