cmd_ethsw.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627
  1. /*
  2. * Copyright 2015 Freescale Semiconductor, Inc.
  3. *
  4. * SPDX-License-Identifier: GPL-2.0+
  5. *
  6. * Ethernet Switch commands
  7. */
  8. #include <common.h>
  9. #include <command.h>
  10. #include <errno.h>
  11. #include <env_flags.h>
  12. #include <ethsw.h>
  13. static const char *ethsw_name;
  14. #define ETHSW_PORT_STATS_HELP "ethsw [port <port_no>] statistics " \
  15. "{ [help] | [clear] } - show an l2 switch port's statistics"
  16. static int ethsw_port_stats_help_key_func(struct ethsw_command_def *parsed_cmd)
  17. {
  18. printf(ETHSW_PORT_STATS_HELP"\n");
  19. return CMD_RET_SUCCESS;
  20. }
  21. #define ETHSW_LEARN_HELP "ethsw [port <port_no>] learning " \
  22. "{ [help] | show | auto | disable } " \
  23. "- enable/disable/show learning configuration on a port"
  24. static int ethsw_learn_help_key_func(struct ethsw_command_def *parsed_cmd)
  25. {
  26. printf(ETHSW_LEARN_HELP"\n");
  27. return CMD_RET_SUCCESS;
  28. }
  29. #define ETHSW_FDB_HELP "ethsw [port <port_no>] [vlan <vid>] fdb " \
  30. "{ [help] | show | flush | { add | del } <mac> } " \
  31. "- Add/delete a mac entry in FDB; use show to see FDB entries; " \
  32. "if vlan <vid> is missing, VID 1 will be used"
  33. static int ethsw_fdb_help_key_func(struct ethsw_command_def *parsed_cmd)
  34. {
  35. printf(ETHSW_FDB_HELP"\n");
  36. return CMD_RET_SUCCESS;
  37. }
  38. static struct keywords_to_function {
  39. enum ethsw_keyword_id cmd_keyword[ETHSW_MAX_CMD_PARAMS];
  40. int cmd_func_offset;
  41. int (*keyword_function)(struct ethsw_command_def *parsed_cmd);
  42. } ethsw_cmd_def[] = {
  43. {
  44. .cmd_keyword = {
  45. ethsw_id_enable,
  46. ethsw_id_key_end,
  47. },
  48. .cmd_func_offset = offsetof(struct ethsw_command_func,
  49. port_enable),
  50. .keyword_function = NULL,
  51. }, {
  52. .cmd_keyword = {
  53. ethsw_id_disable,
  54. ethsw_id_key_end,
  55. },
  56. .cmd_func_offset = offsetof(struct ethsw_command_func,
  57. port_disable),
  58. .keyword_function = NULL,
  59. }, {
  60. .cmd_keyword = {
  61. ethsw_id_show,
  62. ethsw_id_key_end,
  63. },
  64. .cmd_func_offset = offsetof(struct ethsw_command_func,
  65. port_show),
  66. .keyword_function = NULL,
  67. }, {
  68. .cmd_keyword = {
  69. ethsw_id_statistics,
  70. ethsw_id_help,
  71. ethsw_id_key_end,
  72. },
  73. .cmd_func_offset = -1,
  74. .keyword_function = &ethsw_port_stats_help_key_func,
  75. }, {
  76. .cmd_keyword = {
  77. ethsw_id_statistics,
  78. ethsw_id_key_end,
  79. },
  80. .cmd_func_offset = offsetof(struct ethsw_command_func,
  81. port_stats),
  82. .keyword_function = NULL,
  83. }, {
  84. .cmd_keyword = {
  85. ethsw_id_statistics,
  86. ethsw_id_clear,
  87. ethsw_id_key_end,
  88. },
  89. .cmd_func_offset = offsetof(struct ethsw_command_func,
  90. port_stats_clear),
  91. .keyword_function = NULL,
  92. }, {
  93. .cmd_keyword = {
  94. ethsw_id_learning,
  95. ethsw_id_key_end,
  96. },
  97. .cmd_func_offset = -1,
  98. .keyword_function = &ethsw_learn_help_key_func,
  99. }, {
  100. .cmd_keyword = {
  101. ethsw_id_learning,
  102. ethsw_id_help,
  103. ethsw_id_key_end,
  104. },
  105. .cmd_func_offset = -1,
  106. .keyword_function = &ethsw_learn_help_key_func,
  107. }, {
  108. .cmd_keyword = {
  109. ethsw_id_learning,
  110. ethsw_id_show,
  111. ethsw_id_key_end,
  112. },
  113. .cmd_func_offset = offsetof(struct ethsw_command_func,
  114. port_learn_show),
  115. .keyword_function = NULL,
  116. }, {
  117. .cmd_keyword = {
  118. ethsw_id_learning,
  119. ethsw_id_auto,
  120. ethsw_id_key_end,
  121. },
  122. .cmd_func_offset = offsetof(struct ethsw_command_func,
  123. port_learn),
  124. .keyword_function = NULL,
  125. }, {
  126. .cmd_keyword = {
  127. ethsw_id_learning,
  128. ethsw_id_disable,
  129. ethsw_id_key_end,
  130. },
  131. .cmd_func_offset = offsetof(struct ethsw_command_func,
  132. port_learn),
  133. .keyword_function = NULL,
  134. }, {
  135. .cmd_keyword = {
  136. ethsw_id_fdb,
  137. ethsw_id_key_end,
  138. },
  139. .cmd_func_offset = -1,
  140. .keyword_function = &ethsw_fdb_help_key_func,
  141. }, {
  142. .cmd_keyword = {
  143. ethsw_id_fdb,
  144. ethsw_id_help,
  145. ethsw_id_key_end,
  146. },
  147. .cmd_func_offset = -1,
  148. .keyword_function = &ethsw_fdb_help_key_func,
  149. }, {
  150. .cmd_keyword = {
  151. ethsw_id_fdb,
  152. ethsw_id_show,
  153. ethsw_id_key_end,
  154. },
  155. .cmd_func_offset = offsetof(struct ethsw_command_func,
  156. fdb_show),
  157. .keyword_function = NULL,
  158. }, {
  159. .cmd_keyword = {
  160. ethsw_id_fdb,
  161. ethsw_id_flush,
  162. ethsw_id_key_end,
  163. },
  164. .cmd_func_offset = offsetof(struct ethsw_command_func,
  165. fdb_flush),
  166. .keyword_function = NULL,
  167. }, {
  168. .cmd_keyword = {
  169. ethsw_id_fdb,
  170. ethsw_id_add,
  171. ethsw_id_add_del_mac,
  172. ethsw_id_key_end,
  173. },
  174. .cmd_func_offset = offsetof(struct ethsw_command_func,
  175. fdb_entry_add),
  176. .keyword_function = NULL,
  177. }, {
  178. .cmd_keyword = {
  179. ethsw_id_fdb,
  180. ethsw_id_del,
  181. ethsw_id_add_del_mac,
  182. ethsw_id_key_end,
  183. },
  184. .cmd_func_offset = offsetof(struct ethsw_command_func,
  185. fdb_entry_del),
  186. .keyword_function = NULL,
  187. },
  188. };
  189. struct keywords_optional {
  190. int cmd_keyword[ETHSW_MAX_CMD_PARAMS];
  191. } cmd_opt_def[] = {
  192. {
  193. .cmd_keyword = {
  194. ethsw_id_port,
  195. ethsw_id_port_no,
  196. ethsw_id_key_end,
  197. },
  198. }, {
  199. .cmd_keyword = {
  200. ethsw_id_vlan,
  201. ethsw_id_vlan_no,
  202. ethsw_id_key_end,
  203. },
  204. }, {
  205. .cmd_keyword = {
  206. ethsw_id_port,
  207. ethsw_id_port_no,
  208. ethsw_id_vlan,
  209. ethsw_id_vlan_no,
  210. ethsw_id_key_end,
  211. },
  212. },
  213. };
  214. static int keyword_match_gen(enum ethsw_keyword_id key_id, int argc, char
  215. *const argv[], int *argc_nr,
  216. struct ethsw_command_def *parsed_cmd);
  217. static int keyword_match_port(enum ethsw_keyword_id key_id, int argc,
  218. char *const argv[], int *argc_nr,
  219. struct ethsw_command_def *parsed_cmd);
  220. static int keyword_match_vlan(enum ethsw_keyword_id key_id, int argc,
  221. char *const argv[], int *argc_nr,
  222. struct ethsw_command_def *parsed_cmd);
  223. static int keyword_match_mac_addr(enum ethsw_keyword_id key_id, int argc,
  224. char *const argv[], int *argc_nr,
  225. struct ethsw_command_def *parsed_cmd);
  226. /*
  227. * Define properties for each keyword;
  228. * keep the order synced with enum ethsw_keyword_id
  229. */
  230. struct keyword_def {
  231. const char *keyword_name;
  232. int (*match)(enum ethsw_keyword_id key_id, int argc, char *const argv[],
  233. int *argc_nr, struct ethsw_command_def *parsed_cmd);
  234. } keyword[] = {
  235. {
  236. .keyword_name = "help",
  237. .match = &keyword_match_gen,
  238. }, {
  239. .keyword_name = "show",
  240. .match = &keyword_match_gen,
  241. }, {
  242. .keyword_name = "port",
  243. .match = &keyword_match_port
  244. }, {
  245. .keyword_name = "enable",
  246. .match = &keyword_match_gen,
  247. }, {
  248. .keyword_name = "disable",
  249. .match = &keyword_match_gen,
  250. }, {
  251. .keyword_name = "statistics",
  252. .match = &keyword_match_gen,
  253. }, {
  254. .keyword_name = "clear",
  255. .match = &keyword_match_gen,
  256. }, {
  257. .keyword_name = "learning",
  258. .match = &keyword_match_gen,
  259. }, {
  260. .keyword_name = "auto",
  261. .match = &keyword_match_gen,
  262. }, {
  263. .keyword_name = "vlan",
  264. .match = &keyword_match_vlan,
  265. }, {
  266. .keyword_name = "fdb",
  267. .match = &keyword_match_gen,
  268. }, {
  269. .keyword_name = "add",
  270. .match = &keyword_match_mac_addr,
  271. }, {
  272. .keyword_name = "del",
  273. .match = &keyword_match_mac_addr,
  274. }, {
  275. .keyword_name = "flush",
  276. .match = &keyword_match_gen,
  277. },
  278. };
  279. /*
  280. * Function used by an Ethernet Switch driver to set the functions
  281. * that must be called by the parser when an ethsw command is given
  282. */
  283. int ethsw_define_functions(const struct ethsw_command_func *cmd_func)
  284. {
  285. int i;
  286. void **aux_p;
  287. int (*cmd_func_aux)(struct ethsw_command_def *);
  288. if (!cmd_func->ethsw_name)
  289. return -EINVAL;
  290. ethsw_name = cmd_func->ethsw_name;
  291. for (i = 0; i < ARRAY_SIZE(ethsw_cmd_def); i++) {
  292. /*
  293. * get the pointer to the function send by the Ethernet Switch
  294. * driver that corresponds to the proper ethsw command
  295. */
  296. if (ethsw_cmd_def[i].keyword_function)
  297. continue;
  298. aux_p = (void *)cmd_func + ethsw_cmd_def[i].cmd_func_offset;
  299. cmd_func_aux = (int (*)(struct ethsw_command_def *)) *aux_p;
  300. ethsw_cmd_def[i].keyword_function = cmd_func_aux;
  301. }
  302. return 0;
  303. }
  304. /* Generic function used to match a keyword only by a string */
  305. static int keyword_match_gen(enum ethsw_keyword_id key_id, int argc,
  306. char *const argv[], int *argc_nr,
  307. struct ethsw_command_def *parsed_cmd)
  308. {
  309. if (strcmp(argv[*argc_nr], keyword[key_id].keyword_name) == 0) {
  310. parsed_cmd->cmd_to_keywords[*argc_nr] = key_id;
  311. return 1;
  312. }
  313. return 0;
  314. }
  315. /* Function used to match the command's port */
  316. static int keyword_match_port(enum ethsw_keyword_id key_id, int argc,
  317. char *const argv[], int *argc_nr,
  318. struct ethsw_command_def *parsed_cmd)
  319. {
  320. unsigned long val;
  321. if (!keyword_match_gen(key_id, argc, argv, argc_nr, parsed_cmd))
  322. return 0;
  323. if (*argc_nr + 1 >= argc)
  324. return 0;
  325. if (strict_strtoul(argv[*argc_nr + 1], 10, &val) != -EINVAL) {
  326. parsed_cmd->port = val;
  327. (*argc_nr)++;
  328. parsed_cmd->cmd_to_keywords[*argc_nr] = ethsw_id_port_no;
  329. return 1;
  330. }
  331. return 0;
  332. }
  333. /* Function used to match the command's vlan */
  334. static int keyword_match_vlan(enum ethsw_keyword_id key_id, int argc,
  335. char *const argv[], int *argc_nr,
  336. struct ethsw_command_def *parsed_cmd)
  337. {
  338. unsigned long val;
  339. int aux;
  340. if (!keyword_match_gen(key_id, argc, argv, argc_nr, parsed_cmd))
  341. return 0;
  342. if (*argc_nr + 1 >= argc)
  343. return 0;
  344. if (strict_strtoul(argv[*argc_nr + 1], 10, &val) != -EINVAL) {
  345. parsed_cmd->vid = val;
  346. (*argc_nr)++;
  347. parsed_cmd->cmd_to_keywords[*argc_nr] = ethsw_id_vlan_no;
  348. return 1;
  349. }
  350. aux = *argc_nr + 1;
  351. if (keyword_match_gen(ethsw_id_add, argc, argv, &aux, parsed_cmd))
  352. parsed_cmd->cmd_to_keywords[*argc_nr + 1] = ethsw_id_add;
  353. else if (keyword_match_gen(ethsw_id_del, argc, argv, &aux, parsed_cmd))
  354. parsed_cmd->cmd_to_keywords[*argc_nr + 1] = ethsw_id_del;
  355. else
  356. return 0;
  357. if (*argc_nr + 2 >= argc)
  358. return 0;
  359. if (strict_strtoul(argv[*argc_nr + 2], 10, &val) != -EINVAL) {
  360. parsed_cmd->vid = val;
  361. (*argc_nr) += 2;
  362. parsed_cmd->cmd_to_keywords[*argc_nr] = ethsw_id_add_del_no;
  363. return 1;
  364. }
  365. return 0;
  366. }
  367. /* Function used to match the command's MAC address */
  368. static int keyword_match_mac_addr(enum ethsw_keyword_id key_id, int argc,
  369. char *const argv[], int *argc_nr,
  370. struct ethsw_command_def *parsed_cmd)
  371. {
  372. if (!keyword_match_gen(key_id, argc, argv, argc_nr, parsed_cmd))
  373. return 0;
  374. if ((*argc_nr + 1 >= argc) ||
  375. !is_broadcast_ethaddr(parsed_cmd->ethaddr))
  376. return 1;
  377. if (eth_validate_ethaddr_str(argv[*argc_nr + 1])) {
  378. printf("Invalid MAC address: %s\n", argv[*argc_nr + 1]);
  379. return 0;
  380. }
  381. eth_parse_enetaddr(argv[*argc_nr + 1], parsed_cmd->ethaddr);
  382. if (is_broadcast_ethaddr(parsed_cmd->ethaddr)) {
  383. memset(parsed_cmd->ethaddr, 0xFF, sizeof(parsed_cmd->ethaddr));
  384. return 0;
  385. }
  386. parsed_cmd->cmd_to_keywords[*argc_nr + 1] = ethsw_id_add_del_mac;
  387. return 1;
  388. }
  389. /* Finds optional keywords and modifies *argc_va to skip them */
  390. static void cmd_keywords_opt_check(const struct ethsw_command_def *parsed_cmd,
  391. int *argc_val)
  392. {
  393. int i;
  394. int keyw_opt_matched;
  395. int argc_val_max;
  396. int const *cmd_keyw_p;
  397. int const *cmd_keyw_opt_p;
  398. /* remember the best match */
  399. argc_val_max = *argc_val;
  400. /*
  401. * check if our command's optional keywords match the optional
  402. * keywords of an available command
  403. */
  404. for (i = 0; i < ARRAY_SIZE(ethsw_cmd_def); i++) {
  405. keyw_opt_matched = 0;
  406. cmd_keyw_p = &parsed_cmd->cmd_to_keywords[keyw_opt_matched];
  407. cmd_keyw_opt_p = &cmd_opt_def[i].cmd_keyword[keyw_opt_matched];
  408. /*
  409. * increase the number of keywords that
  410. * matched with a command
  411. */
  412. while (keyw_opt_matched + *argc_val <
  413. parsed_cmd->cmd_keywords_nr &&
  414. *cmd_keyw_opt_p != ethsw_id_key_end &&
  415. *(cmd_keyw_p + *argc_val) == *cmd_keyw_opt_p) {
  416. keyw_opt_matched++;
  417. cmd_keyw_p++;
  418. cmd_keyw_opt_p++;
  419. }
  420. /*
  421. * if all our optional command's keywords perfectly match an
  422. * optional pattern, then we can move to the next defined
  423. * keywords in our command; remember the one that matched the
  424. * greatest number of keywords
  425. */
  426. if (keyw_opt_matched + *argc_val <=
  427. parsed_cmd->cmd_keywords_nr &&
  428. *cmd_keyw_opt_p == ethsw_id_key_end &&
  429. *argc_val + keyw_opt_matched > argc_val_max)
  430. argc_val_max = *argc_val + keyw_opt_matched;
  431. }
  432. *argc_val = argc_val_max;
  433. }
  434. /*
  435. * Finds the function to call based on keywords and
  436. * modifies *argc_va to skip them
  437. */
  438. static void cmd_keywords_check(struct ethsw_command_def *parsed_cmd,
  439. int *argc_val)
  440. {
  441. int i;
  442. int keyw_matched;
  443. int *cmd_keyw_p;
  444. int *cmd_keyw_def_p;
  445. /*
  446. * check if our command's keywords match the
  447. * keywords of an available command
  448. */
  449. for (i = 0; i < ARRAY_SIZE(ethsw_cmd_def); i++) {
  450. keyw_matched = 0;
  451. cmd_keyw_p = &parsed_cmd->cmd_to_keywords[keyw_matched];
  452. cmd_keyw_def_p = &ethsw_cmd_def[i].cmd_keyword[keyw_matched];
  453. /*
  454. * increase the number of keywords that
  455. * matched with a command
  456. */
  457. while (keyw_matched + *argc_val < parsed_cmd->cmd_keywords_nr &&
  458. *cmd_keyw_def_p != ethsw_id_key_end &&
  459. *(cmd_keyw_p + *argc_val) == *cmd_keyw_def_p) {
  460. keyw_matched++;
  461. cmd_keyw_p++;
  462. cmd_keyw_def_p++;
  463. }
  464. /*
  465. * if all our command's keywords perfectly match an
  466. * available command, then we get the function we need to call
  467. * to configure the Ethernet Switch
  468. */
  469. if (keyw_matched && keyw_matched + *argc_val ==
  470. parsed_cmd->cmd_keywords_nr &&
  471. *cmd_keyw_def_p == ethsw_id_key_end) {
  472. *argc_val += keyw_matched;
  473. parsed_cmd->cmd_function =
  474. ethsw_cmd_def[i].keyword_function;
  475. return;
  476. }
  477. }
  478. }
  479. /* find all the keywords in the command */
  480. static int keywords_find(int argc, char * const argv[],
  481. struct ethsw_command_def *parsed_cmd)
  482. {
  483. int i;
  484. int j;
  485. int argc_val;
  486. int rc = CMD_RET_SUCCESS;
  487. for (i = 1; i < argc; i++) {
  488. for (j = 0; j < ethsw_id_count; j++) {
  489. if (keyword[j].match(j, argc, argv, &i, parsed_cmd))
  490. break;
  491. }
  492. }
  493. /* if there is no keyword match for a word, the command is invalid */
  494. for (i = 1; i < argc; i++)
  495. if (parsed_cmd->cmd_to_keywords[i] == ethsw_id_key_end)
  496. rc = CMD_RET_USAGE;
  497. parsed_cmd->cmd_keywords_nr = argc;
  498. argc_val = 1;
  499. /* get optional parameters first */
  500. cmd_keywords_opt_check(parsed_cmd, &argc_val);
  501. if (argc_val == parsed_cmd->cmd_keywords_nr)
  502. return CMD_RET_USAGE;
  503. /*
  504. * check the keywords and if a match is found,
  505. * get the function to call
  506. */
  507. cmd_keywords_check(parsed_cmd, &argc_val);
  508. /* error if not all commands' parameters were matched */
  509. if (argc_val == parsed_cmd->cmd_keywords_nr) {
  510. if (!parsed_cmd->cmd_function) {
  511. printf("Command not available for: %s\n", ethsw_name);
  512. rc = CMD_RET_FAILURE;
  513. }
  514. } else {
  515. rc = CMD_RET_USAGE;
  516. }
  517. return rc;
  518. }
  519. static void command_def_init(struct ethsw_command_def *parsed_cmd)
  520. {
  521. int i;
  522. for (i = 0; i < ETHSW_MAX_CMD_PARAMS; i++)
  523. parsed_cmd->cmd_to_keywords[i] = ethsw_id_key_end;
  524. parsed_cmd->port = ETHSW_CMD_PORT_ALL;
  525. parsed_cmd->vid = ETHSW_CMD_VLAN_ALL;
  526. parsed_cmd->cmd_function = NULL;
  527. /* We initialize the MAC address with the Broadcast address */
  528. memset(parsed_cmd->ethaddr, 0xff, sizeof(parsed_cmd->ethaddr));
  529. }
  530. /* function to interpret commands starting with "ethsw " */
  531. static int do_ethsw(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  532. {
  533. struct ethsw_command_def parsed_cmd;
  534. int rc = CMD_RET_SUCCESS;
  535. if (argc == 1 || argc >= ETHSW_MAX_CMD_PARAMS)
  536. return CMD_RET_USAGE;
  537. command_def_init(&parsed_cmd);
  538. rc = keywords_find(argc, argv, &parsed_cmd);
  539. if (rc == CMD_RET_SUCCESS)
  540. rc = parsed_cmd.cmd_function(&parsed_cmd);
  541. return rc;
  542. }
  543. #define ETHSW_PORT_CONF_HELP "[port <port_no>] { enable | disable | show } " \
  544. "- enable/disable a port; show shows a port's configuration"
  545. U_BOOT_CMD(ethsw, ETHSW_MAX_CMD_PARAMS, 0, do_ethsw,
  546. "Ethernet l2 switch commands",
  547. ETHSW_PORT_CONF_HELP"\n"
  548. ETHSW_PORT_STATS_HELP"\n"
  549. ETHSW_LEARN_HELP"\n"
  550. ETHSW_FDB_HELP"\n"
  551. );