net/sfc: support counters in tunnel offload jump rules
Such a counter will only report the number of hits, which is actually a sum of two contributions (the JUMP rule's own counter + indirect increments issued by counters of the associated GROUP rules. Signed-off-by: Ivan Malov <ivan.malov@oktetlabs.ru> Reviewed-by: Andrew Rybchenko <andrew.rybchenko@oktetlabs.ru>
This commit is contained in:
parent
8cd7725169
commit
9df2d8f5cc
@ -2993,6 +2993,8 @@ sfc_flow_start(struct sfc_adapter *sa)
|
||||
|
||||
SFC_ASSERT(sfc_adapter_is_locked(sa));
|
||||
|
||||
sfc_flow_tunnel_reset_hit_counters(sa);
|
||||
|
||||
TAILQ_FOREACH(flow, &sa->flow_list, entries) {
|
||||
rc = sfc_flow_insert(sa, flow, NULL);
|
||||
if (rc != 0)
|
||||
|
@ -88,6 +88,8 @@ sfc_flow_tunnel_detect_jump_rule(struct sfc_adapter *sa,
|
||||
}
|
||||
|
||||
switch (actions->type) {
|
||||
case RTE_FLOW_ACTION_TYPE_COUNT:
|
||||
break;
|
||||
case RTE_FLOW_ACTION_TYPE_MARK:
|
||||
if (action_mark == NULL) {
|
||||
action_mark = actions->conf;
|
||||
@ -460,3 +462,19 @@ fail:
|
||||
RTE_FLOW_ERROR_TYPE_UNSPECIFIED, NULL,
|
||||
"tunnel offload: get_restore_info failed");
|
||||
}
|
||||
|
||||
void
|
||||
sfc_flow_tunnel_reset_hit_counters(struct sfc_adapter *sa)
|
||||
{
|
||||
unsigned int i;
|
||||
|
||||
SFC_ASSERT(sfc_adapter_is_locked(sa));
|
||||
SFC_ASSERT(sa->state != SFC_ETHDEV_STARTED);
|
||||
|
||||
for (i = 0; i < RTE_DIM(sa->flow_tunnels); ++i) {
|
||||
struct sfc_flow_tunnel *ft = &sa->flow_tunnels[i];
|
||||
|
||||
ft->reset_jump_hit_counter = 0;
|
||||
ft->group_hit_counter = 0;
|
||||
}
|
||||
}
|
||||
|
@ -63,6 +63,9 @@ struct sfc_flow_tunnel {
|
||||
struct rte_flow_item_mark item_mark_v;
|
||||
struct rte_flow_item_mark item_mark_m;
|
||||
struct rte_flow_item item;
|
||||
|
||||
uint64_t reset_jump_hit_counter;
|
||||
uint64_t group_hit_counter;
|
||||
};
|
||||
|
||||
struct sfc_adapter;
|
||||
@ -106,6 +109,8 @@ int sfc_flow_tunnel_get_restore_info(struct rte_eth_dev *dev,
|
||||
struct rte_flow_restore_info *info,
|
||||
struct rte_flow_error *err);
|
||||
|
||||
void sfc_flow_tunnel_reset_hit_counters(struct sfc_adapter *sa);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
@ -737,6 +737,8 @@ sfc_mae_action_set_add(struct sfc_adapter *sa,
|
||||
const struct rte_flow_action actions[],
|
||||
efx_mae_actions_t *spec,
|
||||
struct sfc_mae_encap_header *encap_header,
|
||||
uint64_t *ft_group_hit_counter,
|
||||
struct sfc_flow_tunnel *ft,
|
||||
unsigned int n_counters,
|
||||
struct sfc_mae_action_set **action_setp)
|
||||
{
|
||||
@ -763,6 +765,16 @@ sfc_mae_action_set_add(struct sfc_adapter *sa,
|
||||
return ENOMEM;
|
||||
}
|
||||
|
||||
for (i = 0; i < n_counters; ++i) {
|
||||
action_set->counters[i].rte_id_valid = B_FALSE;
|
||||
action_set->counters[i].mae_id.id =
|
||||
EFX_MAE_RSRC_ID_INVALID;
|
||||
|
||||
action_set->counters[i].ft_group_hit_counter =
|
||||
ft_group_hit_counter;
|
||||
action_set->counters[i].ft = ft;
|
||||
}
|
||||
|
||||
for (action = actions, i = 0;
|
||||
action->type != RTE_FLOW_ACTION_TYPE_END && i < n_counters;
|
||||
++action) {
|
||||
@ -773,8 +785,7 @@ sfc_mae_action_set_add(struct sfc_adapter *sa,
|
||||
|
||||
conf = action->conf;
|
||||
|
||||
action_set->counters[i].mae_id.id =
|
||||
EFX_MAE_RSRC_ID_INVALID;
|
||||
action_set->counters[i].rte_id_valid = B_TRUE;
|
||||
action_set->counters[i].rte_id = conf->id;
|
||||
i++;
|
||||
}
|
||||
@ -3499,10 +3510,12 @@ sfc_mae_rule_parse_actions(struct sfc_adapter *sa,
|
||||
{
|
||||
struct sfc_mae_encap_header *encap_header = NULL;
|
||||
struct sfc_mae_actions_bundle bundle = {0};
|
||||
struct sfc_flow_tunnel *counter_ft = NULL;
|
||||
uint64_t *ft_group_hit_counter = NULL;
|
||||
const struct rte_flow_action *action;
|
||||
struct sfc_mae *mae = &sa->mae;
|
||||
unsigned int n_count = 0;
|
||||
efx_mae_actions_t *spec;
|
||||
unsigned int n_count;
|
||||
int rc;
|
||||
|
||||
rte_errno = 0;
|
||||
@ -3517,11 +3530,31 @@ sfc_mae_rule_parse_actions(struct sfc_adapter *sa,
|
||||
if (rc != 0)
|
||||
goto fail_action_set_spec_init;
|
||||
|
||||
for (action = actions;
|
||||
action->type != RTE_FLOW_ACTION_TYPE_END; ++action) {
|
||||
if (action->type == RTE_FLOW_ACTION_TYPE_COUNT)
|
||||
++n_count;
|
||||
}
|
||||
|
||||
if (spec_mae->ft_rule_type == SFC_FT_RULE_GROUP) {
|
||||
/* JUMP rules don't decapsulate packets. GROUP rules do. */
|
||||
rc = efx_mae_action_set_populate_decap(spec);
|
||||
if (rc != 0)
|
||||
goto fail_enforce_ft_decap;
|
||||
|
||||
if (n_count == 0 && sfc_mae_counter_stream_enabled(sa)) {
|
||||
/*
|
||||
* The user opted not to use action COUNT in this rule,
|
||||
* but the counter should be enabled implicitly because
|
||||
* packets hitting this rule contribute to the tunnel's
|
||||
* total number of hits. See sfc_mae_counter_get().
|
||||
*/
|
||||
rc = efx_mae_action_set_populate_count(spec);
|
||||
if (rc != 0)
|
||||
goto fail_enforce_ft_count;
|
||||
|
||||
n_count = 1;
|
||||
}
|
||||
}
|
||||
|
||||
/* Cleanup after previous encap. header bounce buffer usage. */
|
||||
@ -3547,7 +3580,6 @@ sfc_mae_rule_parse_actions(struct sfc_adapter *sa,
|
||||
if (rc != 0)
|
||||
goto fail_process_encap_header;
|
||||
|
||||
n_count = efx_mae_action_set_get_nb_count(spec);
|
||||
if (n_count > 1) {
|
||||
rc = ENOTSUP;
|
||||
sfc_err(sa, "too many count actions requested: %u", n_count);
|
||||
@ -3562,6 +3594,8 @@ sfc_mae_rule_parse_actions(struct sfc_adapter *sa,
|
||||
rc = sfc_mae_rule_parse_action_pf_vf(sa, NULL, spec);
|
||||
if (rc != 0)
|
||||
goto fail_workaround_jump_delivery;
|
||||
|
||||
counter_ft = spec_mae->ft;
|
||||
break;
|
||||
case SFC_FT_RULE_GROUP:
|
||||
/*
|
||||
@ -3571,6 +3605,8 @@ sfc_mae_rule_parse_actions(struct sfc_adapter *sa,
|
||||
* MARK above, so don't check the return value here.
|
||||
*/
|
||||
(void)efx_mae_action_set_populate_mark(spec, 0);
|
||||
|
||||
ft_group_hit_counter = &spec_mae->ft->group_hit_counter;
|
||||
break;
|
||||
default:
|
||||
SFC_ASSERT(B_FALSE);
|
||||
@ -3584,7 +3620,8 @@ sfc_mae_rule_parse_actions(struct sfc_adapter *sa,
|
||||
return 0;
|
||||
}
|
||||
|
||||
rc = sfc_mae_action_set_add(sa, actions, spec, encap_header, n_count,
|
||||
rc = sfc_mae_action_set_add(sa, actions, spec, encap_header,
|
||||
ft_group_hit_counter, counter_ft, n_count,
|
||||
&spec_mae->action_set);
|
||||
if (rc != 0)
|
||||
goto fail_action_set_add;
|
||||
@ -3600,6 +3637,7 @@ fail_process_encap_header:
|
||||
fail_rule_parse_action:
|
||||
efx_mae_action_set_spec_fini(sa->nic, spec);
|
||||
|
||||
fail_enforce_ft_count:
|
||||
fail_enforce_ft_decap:
|
||||
fail_action_set_spec_init:
|
||||
if (rc > 0 && rte_errno == 0) {
|
||||
@ -3747,6 +3785,11 @@ sfc_mae_flow_insert(struct sfc_adapter *sa,
|
||||
goto fail_outer_rule_enable;
|
||||
}
|
||||
|
||||
if (spec_mae->ft_rule_type == SFC_FT_RULE_JUMP) {
|
||||
spec_mae->ft->reset_jump_hit_counter =
|
||||
spec_mae->ft->group_hit_counter;
|
||||
}
|
||||
|
||||
if (action_set == NULL) {
|
||||
sfc_dbg(sa, "enabled flow=%p (no AR)", flow);
|
||||
return 0;
|
||||
@ -3846,7 +3889,8 @@ sfc_mae_query_counter(struct sfc_adapter *sa,
|
||||
for (i = 0; i < action_set->n_counters; i++) {
|
||||
/*
|
||||
* Get the first available counter of the flow rule if
|
||||
* counter ID is not specified.
|
||||
* counter ID is not specified, provided that this
|
||||
* counter is not an automatic (implicit) one.
|
||||
*/
|
||||
if (conf != NULL && action_set->counters[i].rte_id != conf->id)
|
||||
continue;
|
||||
@ -3864,7 +3908,7 @@ sfc_mae_query_counter(struct sfc_adapter *sa,
|
||||
|
||||
return rte_flow_error_set(error, ENOENT,
|
||||
RTE_FLOW_ERROR_TYPE_ACTION, action,
|
||||
"No such flow rule action count ID");
|
||||
"no such flow rule action or such count ID");
|
||||
}
|
||||
|
||||
int
|
||||
|
@ -62,6 +62,13 @@ struct sfc_mae_counter_id {
|
||||
efx_counter_t mae_id;
|
||||
/* ID of a counter in RTE */
|
||||
uint32_t rte_id;
|
||||
/* RTE counter ID validity status */
|
||||
bool rte_id_valid;
|
||||
|
||||
/* Flow Tunnel (FT) GROUP hit counter (or NULL) */
|
||||
uint64_t *ft_group_hit_counter;
|
||||
/* Flow Tunnel (FT) context (for JUMP rules; otherwise, NULL) */
|
||||
struct sfc_flow_tunnel *ft;
|
||||
};
|
||||
|
||||
/** Action set registry entry */
|
||||
@ -101,6 +108,8 @@ struct sfc_mae_counter {
|
||||
uint32_t generation_count;
|
||||
union sfc_pkts_bytes value;
|
||||
union sfc_pkts_bytes reset;
|
||||
|
||||
uint64_t *ft_group_hit_counter;
|
||||
};
|
||||
|
||||
struct sfc_mae_counters_xstats {
|
||||
|
@ -99,6 +99,8 @@ sfc_mae_counter_enable(struct sfc_adapter *sa,
|
||||
&p->value.pkts_bytes.int128, __ATOMIC_RELAXED);
|
||||
p->generation_count = generation_count;
|
||||
|
||||
p->ft_group_hit_counter = counterp->ft_group_hit_counter;
|
||||
|
||||
/*
|
||||
* The flag is set at the very end of add operation and reset
|
||||
* at the beginning of delete operation. Release ordering is
|
||||
@ -210,6 +212,14 @@ sfc_mae_counter_increment(struct sfc_adapter *sa,
|
||||
__atomic_store(&p->value.pkts_bytes,
|
||||
&cnt_val.pkts_bytes, __ATOMIC_RELAXED);
|
||||
|
||||
if (p->ft_group_hit_counter != NULL) {
|
||||
uint64_t ft_group_hit_counter;
|
||||
|
||||
ft_group_hit_counter = *p->ft_group_hit_counter + pkts;
|
||||
__atomic_store_n(p->ft_group_hit_counter, ft_group_hit_counter,
|
||||
__ATOMIC_RELAXED);
|
||||
}
|
||||
|
||||
sfc_info(sa, "update MAE counter #%u: pkts+%" PRIu64 "=%" PRIu64
|
||||
", bytes+%" PRIu64 "=%" PRIu64, mae_counter_id,
|
||||
pkts, cnt_val.pkts, bytes, cnt_val.bytes);
|
||||
@ -799,6 +809,8 @@ sfc_mae_counter_get(struct sfc_mae_counters *counters,
|
||||
const struct sfc_mae_counter_id *counter,
|
||||
struct rte_flow_query_count *data)
|
||||
{
|
||||
struct sfc_flow_tunnel *ft = counter->ft;
|
||||
uint64_t non_reset_jump_hit_counter;
|
||||
struct sfc_mae_counter *p;
|
||||
union sfc_pkts_bytes value;
|
||||
|
||||
@ -814,14 +826,35 @@ sfc_mae_counter_get(struct sfc_mae_counters *counters,
|
||||
__ATOMIC_RELAXED);
|
||||
|
||||
data->hits_set = 1;
|
||||
data->bytes_set = 1;
|
||||
data->hits = value.pkts - p->reset.pkts;
|
||||
|
||||
if (ft != NULL) {
|
||||
data->hits += ft->group_hit_counter;
|
||||
non_reset_jump_hit_counter = data->hits;
|
||||
data->hits -= ft->reset_jump_hit_counter;
|
||||
} else {
|
||||
data->bytes_set = 1;
|
||||
data->bytes = value.bytes - p->reset.bytes;
|
||||
}
|
||||
|
||||
if (data->reset != 0) {
|
||||
if (ft != NULL) {
|
||||
ft->reset_jump_hit_counter = non_reset_jump_hit_counter;
|
||||
} else {
|
||||
p->reset.pkts = value.pkts;
|
||||
p->reset.bytes = value.bytes;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
bool
|
||||
sfc_mae_counter_stream_enabled(struct sfc_adapter *sa)
|
||||
{
|
||||
if ((sa->counter_rxq.state & SFC_COUNTER_RXQ_INITIALIZED) == 0 ||
|
||||
sfc_get_service_lcore(SOCKET_ID_ANY) == RTE_MAX_LCORE)
|
||||
return B_FALSE;
|
||||
else
|
||||
return B_TRUE;
|
||||
}
|
||||
|
@ -52,6 +52,9 @@ int sfc_mae_counter_get(struct sfc_mae_counters *counters,
|
||||
int sfc_mae_counter_start(struct sfc_adapter *sa);
|
||||
void sfc_mae_counter_stop(struct sfc_adapter *sa);
|
||||
|
||||
/* Check whether MAE Counter-on-Queue (CoQ) prerequisites are satisfied */
|
||||
bool sfc_mae_counter_stream_enabled(struct sfc_adapter *sa);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
Loading…
x
Reference in New Issue
Block a user