// SPDX-License-Identifier: GPL-2.0+ #include #include #include #include "dcb.h" #include "utils.h" static void dcb_pfc_help_set(void) { fprintf(stderr, "Usage: dcb pfc set dev STRING\n" " [ prio-pfc PFC-MAP ]\n" " [ macsec-bypass { on | off } ]\n" " [ delay INTEGER ]\n" "\n" " where PFC-MAP := [ PFC-MAP ] PFC-MAPPING\n" " PFC-MAPPING := { all | TC }:PFC\n" " TC := { 0 .. 7 }\n" " PFC := { on | off }\n" "\n" ); } static void dcb_pfc_help_show(void) { fprintf(stderr, "Usage: dcb [ -s ] pfc show dev STRING\n" " [ pfc-cap ] [ prio-pfc ] [ macsec-bypass ]\n" " [ delay ] [ requests ] [ indications ]\n" "\n" ); } static void dcb_pfc_help(void) { fprintf(stderr, "Usage: dcb pfc help\n" "\n" ); dcb_pfc_help_show(); dcb_pfc_help_set(); } static void dcb_pfc_to_array(__u8 array[IEEE_8021QAZ_MAX_TCS], __u8 pfc_en) { int i; for (i = 0; i < IEEE_8021QAZ_MAX_TCS; i++) array[i] = !!(pfc_en & (1 << i)); } static void dcb_pfc_from_array(__u8 array[IEEE_8021QAZ_MAX_TCS], __u8 *pfc_en_p) { __u8 pfc_en = 0; int i; for (i = 0; i < IEEE_8021QAZ_MAX_TCS; i++) { if (array[i]) pfc_en |= 1 << i; } *pfc_en_p = pfc_en; } static int dcb_pfc_parse_mapping_prio_pfc(__u32 key, char *value, void *data) { struct ieee_pfc *pfc = data; __u8 pfc_en[IEEE_8021QAZ_MAX_TCS]; bool enabled; int ret; dcb_pfc_to_array(pfc_en, pfc->pfc_en); enabled = parse_on_off("PFC", value, &ret); if (ret) return ret; ret = dcb_parse_mapping("PRIO", key, IEEE_8021QAZ_MAX_TCS - 1, "PFC", enabled, -1, dcb_set_u8, pfc_en); if (ret) return ret; dcb_pfc_from_array(pfc_en, &pfc->pfc_en); return 0; } static void dcb_pfc_print_pfc_cap(const struct ieee_pfc *pfc) { print_uint(PRINT_ANY, "pfc_cap", "pfc-cap %d ", pfc->pfc_cap); } static void dcb_pfc_print_macsec_bypass(const struct ieee_pfc *pfc) { print_on_off(PRINT_ANY, "macsec_bypass", "macsec-bypass %s ", pfc->mbc); } static void dcb_pfc_print_delay(const struct ieee_pfc *pfc) { print_uint(PRINT_ANY, "delay", "delay %d ", pfc->delay); } static void dcb_pfc_print_prio_pfc(const struct ieee_pfc *pfc) { __u8 pfc_en[IEEE_8021QAZ_MAX_TCS]; dcb_pfc_to_array(pfc_en, pfc->pfc_en); dcb_print_named_array("prio_pfc", "prio-pfc", pfc_en, ARRAY_SIZE(pfc_en), &dcb_print_array_on_off); } static void dcb_pfc_print_requests(const struct ieee_pfc *pfc) { open_json_array(PRINT_JSON, "requests"); print_string(PRINT_FP, NULL, "requests ", NULL); dcb_print_array_u64(pfc->requests, ARRAY_SIZE(pfc->requests)); close_json_array(PRINT_JSON, "requests"); } static void dcb_pfc_print_indications(const struct ieee_pfc *pfc) { open_json_array(PRINT_JSON, "indications"); print_string(PRINT_FP, NULL, "indications ", NULL); dcb_print_array_u64(pfc->indications, ARRAY_SIZE(pfc->indications)); close_json_array(PRINT_JSON, "indications"); } static void dcb_pfc_print(const struct dcb *dcb, const struct ieee_pfc *pfc) { dcb_pfc_print_pfc_cap(pfc); dcb_pfc_print_macsec_bypass(pfc); dcb_pfc_print_delay(pfc); print_nl(); dcb_pfc_print_prio_pfc(pfc); print_nl(); if (dcb->stats) { dcb_pfc_print_requests(pfc); print_nl(); dcb_pfc_print_indications(pfc); print_nl(); } } static int dcb_pfc_get(struct dcb *dcb, const char *dev, struct ieee_pfc *pfc) { return dcb_get_attribute(dcb, dev, DCB_ATTR_IEEE_PFC, pfc, sizeof(*pfc)); } static int dcb_pfc_set(struct dcb *dcb, const char *dev, const struct ieee_pfc *pfc) { return dcb_set_attribute(dcb, dev, DCB_ATTR_IEEE_PFC, pfc, sizeof(*pfc)); } static int dcb_cmd_pfc_set(struct dcb *dcb, const char *dev, int argc, char **argv) { struct ieee_pfc pfc; int ret; if (!argc) { dcb_pfc_help_set(); return 0; } ret = dcb_pfc_get(dcb, dev, &pfc); if (ret) return ret; do { if (matches(*argv, "help") == 0) { dcb_pfc_help_set(); return 0; } else if (matches(*argv, "prio-pfc") == 0) { NEXT_ARG(); ret = parse_mapping(&argc, &argv, true, &dcb_pfc_parse_mapping_prio_pfc, &pfc); if (ret) { fprintf(stderr, "Invalid pfc mapping %s\n", *argv); return ret; } continue; } else if (matches(*argv, "macsec-bypass") == 0) { NEXT_ARG(); pfc.mbc = parse_on_off("macsec-bypass", *argv, &ret); if (ret) return ret; } else if (matches(*argv, "delay") == 0) { NEXT_ARG(); /* Do not support the size notations for delay. * Delay is specified in "bit times", not bits, so * it is not applicable. At the same time it would * be confusing that 10Kbit does not mean 10240, * but 1280. */ if (get_u16(&pfc.delay, *argv, 0)) { fprintf(stderr, "Invalid delay `%s', expected an integer 0..65535\n", *argv); return -EINVAL; } } else { fprintf(stderr, "What is \"%s\"?\n", *argv); dcb_pfc_help_set(); return -EINVAL; } NEXT_ARG_FWD(); } while (argc > 0); return dcb_pfc_set(dcb, dev, &pfc); } static int dcb_cmd_pfc_show(struct dcb *dcb, const char *dev, int argc, char **argv) { struct ieee_pfc pfc; int ret; ret = dcb_pfc_get(dcb, dev, &pfc); if (ret) return ret; open_json_object(NULL); if (!argc) { dcb_pfc_print(dcb, &pfc); goto out; } do { if (matches(*argv, "help") == 0) { dcb_pfc_help_show(); return 0; } else if (matches(*argv, "prio-pfc") == 0) { dcb_pfc_print_prio_pfc(&pfc); print_nl(); } else if (matches(*argv, "pfc-cap") == 0) { dcb_pfc_print_pfc_cap(&pfc); print_nl(); } else if (matches(*argv, "macsec-bypass") == 0) { dcb_pfc_print_macsec_bypass(&pfc); print_nl(); } else if (matches(*argv, "delay") == 0) { dcb_pfc_print_delay(&pfc); print_nl(); } else if (matches(*argv, "requests") == 0) { dcb_pfc_print_requests(&pfc); print_nl(); } else if (matches(*argv, "indications") == 0) { dcb_pfc_print_indications(&pfc); print_nl(); } else { fprintf(stderr, "What is \"%s\"?\n", *argv); dcb_pfc_help_show(); return -EINVAL; } NEXT_ARG_FWD(); } while (argc > 0); out: close_json_object(); return 0; } int dcb_cmd_pfc(struct dcb *dcb, int argc, char **argv) { if (!argc || matches(*argv, "help") == 0) { dcb_pfc_help(); return 0; } else if (matches(*argv, "show") == 0) { NEXT_ARG_FWD(); return dcb_cmd_parse_dev(dcb, argc, argv, dcb_cmd_pfc_show, dcb_pfc_help_show); } else if (matches(*argv, "set") == 0) { NEXT_ARG_FWD(); return dcb_cmd_parse_dev(dcb, argc, argv, dcb_cmd_pfc_set, dcb_pfc_help_set); } else { fprintf(stderr, "What is \"%s\"?\n", *argv); dcb_pfc_help(); return -EINVAL; } }