Commit d4ece08f authored by Patrisious Haddad's avatar Patrisious Haddad Committed by Leon Romanovsky

net/mlx5: Maintain fs_id xarray per MACsec device inside macsec steering

Remove fs_id from the MACsec SA, since it has no real usage there and
instead maintain with the MACsec steering data inside the core.

Downstream patches requires this change to facilitate IB driver accesses
to the fs_ids to avoid RoCE MACsec dependency on EN driver.
Signed-off-by: default avatarPatrisious Haddad <phaddad@nvidia.com>
Signed-off-by: default avatarLeon Romanovsky <leonro@nvidia.com>
parent 7e2304f5
...@@ -65,8 +65,6 @@ struct mlx5e_macsec_sa { ...@@ -65,8 +65,6 @@ struct mlx5e_macsec_sa {
ssci_t ssci; ssci_t ssci;
salt_t salt; salt_t salt;
struct rhash_head hash;
u32 fs_id;
union mlx5_macsec_rule *macsec_rule; union mlx5_macsec_rule *macsec_rule;
struct rcu_head rcu_head; struct rcu_head rcu_head;
struct mlx5e_macsec_epn_state epn_state; struct mlx5e_macsec_epn_state epn_state;
...@@ -105,14 +103,6 @@ struct mlx5e_macsec_aso { ...@@ -105,14 +103,6 @@ struct mlx5e_macsec_aso {
u32 pdn; u32 pdn;
}; };
static const struct rhashtable_params rhash_sci = {
.key_len = sizeof_field(struct mlx5e_macsec_sa, sci),
.key_offset = offsetof(struct mlx5e_macsec_sa, sci),
.head_offset = offsetof(struct mlx5e_macsec_sa, hash),
.automatic_shrinking = true,
.min_size = 1,
};
struct mlx5e_macsec_device { struct mlx5e_macsec_device {
const struct net_device *netdev; const struct net_device *netdev;
struct mlx5e_macsec_sa *tx_sa[MACSEC_NUM_AN]; struct mlx5e_macsec_sa *tx_sa[MACSEC_NUM_AN];
...@@ -126,9 +116,6 @@ struct mlx5e_macsec { ...@@ -126,9 +116,6 @@ struct mlx5e_macsec {
int num_of_devices; int num_of_devices;
struct mutex lock; /* Protects mlx5e_macsec internal contexts */ struct mutex lock; /* Protects mlx5e_macsec internal contexts */
/* Tx sci -> fs id mapping handling */
struct rhashtable sci_hash; /* sci -> mlx5e_macsec_sa */
/* Rx fs_id -> rx_sc mapping */ /* Rx fs_id -> rx_sc mapping */
struct xarray sc_xarray; struct xarray sc_xarray;
...@@ -325,29 +312,23 @@ static void mlx5e_macsec_destroy_object(struct mlx5_core_dev *mdev, u32 macsec_o ...@@ -325,29 +312,23 @@ static void mlx5e_macsec_destroy_object(struct mlx5_core_dev *mdev, u32 macsec_o
static void mlx5e_macsec_cleanup_sa(struct mlx5e_macsec *macsec, static void mlx5e_macsec_cleanup_sa(struct mlx5e_macsec *macsec,
struct mlx5e_macsec_sa *sa, struct mlx5e_macsec_sa *sa,
bool is_tx) bool is_tx, struct net_device *netdev, u32 fs_id)
{ {
int action = (is_tx) ? MLX5_ACCEL_MACSEC_ACTION_ENCRYPT : int action = (is_tx) ? MLX5_ACCEL_MACSEC_ACTION_ENCRYPT :
MLX5_ACCEL_MACSEC_ACTION_DECRYPT; MLX5_ACCEL_MACSEC_ACTION_DECRYPT;
if ((is_tx) && sa->fs_id) {
/* Make sure ongoing datapath readers sees a valid SA */
rhashtable_remove_fast(&macsec->sci_hash, &sa->hash, rhash_sci);
sa->fs_id = 0;
}
if (!sa->macsec_rule) if (!sa->macsec_rule)
return; return;
mlx5_macsec_fs_del_rule(macsec->mdev->macsec_fs, sa->macsec_rule, action); mlx5_macsec_fs_del_rule(macsec->mdev->macsec_fs, sa->macsec_rule, action, netdev,
fs_id);
mlx5e_macsec_destroy_object(macsec->mdev, sa->macsec_obj_id); mlx5e_macsec_destroy_object(macsec->mdev, sa->macsec_obj_id);
sa->macsec_rule = NULL; sa->macsec_rule = NULL;
} }
static int mlx5e_macsec_init_sa(struct macsec_context *ctx, static int mlx5e_macsec_init_sa(struct macsec_context *ctx,
struct mlx5e_macsec_sa *sa, struct mlx5e_macsec_sa *sa,
bool encrypt, bool encrypt, bool is_tx, u32 *fs_id)
bool is_tx)
{ {
struct mlx5e_priv *priv = macsec_netdev_priv(ctx->netdev); struct mlx5e_priv *priv = macsec_netdev_priv(ctx->netdev);
struct mlx5e_macsec *macsec = priv->macsec; struct mlx5e_macsec *macsec = priv->macsec;
...@@ -382,7 +363,7 @@ static int mlx5e_macsec_init_sa(struct macsec_context *ctx, ...@@ -382,7 +363,7 @@ static int mlx5e_macsec_init_sa(struct macsec_context *ctx,
rule_attrs.action = (is_tx) ? MLX5_ACCEL_MACSEC_ACTION_ENCRYPT : rule_attrs.action = (is_tx) ? MLX5_ACCEL_MACSEC_ACTION_ENCRYPT :
MLX5_ACCEL_MACSEC_ACTION_DECRYPT; MLX5_ACCEL_MACSEC_ACTION_DECRYPT;
macsec_rule = mlx5_macsec_fs_add_rule(mdev->macsec_fs, ctx, &rule_attrs, &sa->fs_id); macsec_rule = mlx5_macsec_fs_add_rule(mdev->macsec_fs, ctx, &rule_attrs, fs_id);
if (!macsec_rule) { if (!macsec_rule) {
err = -ENOMEM; err = -ENOMEM;
goto destroy_macsec_object; goto destroy_macsec_object;
...@@ -390,16 +371,8 @@ static int mlx5e_macsec_init_sa(struct macsec_context *ctx, ...@@ -390,16 +371,8 @@ static int mlx5e_macsec_init_sa(struct macsec_context *ctx,
sa->macsec_rule = macsec_rule; sa->macsec_rule = macsec_rule;
if (is_tx) {
err = rhashtable_insert_fast(&macsec->sci_hash, &sa->hash, rhash_sci);
if (err)
goto destroy_macsec_object_and_rule;
}
return 0; return 0;
destroy_macsec_object_and_rule:
mlx5e_macsec_cleanup_sa(macsec, sa, is_tx);
destroy_macsec_object: destroy_macsec_object:
mlx5e_macsec_destroy_object(mdev, sa->macsec_obj_id); mlx5e_macsec_destroy_object(mdev, sa->macsec_obj_id);
...@@ -421,7 +394,7 @@ mlx5e_macsec_get_rx_sc_from_sc_list(const struct list_head *list, sci_t sci) ...@@ -421,7 +394,7 @@ mlx5e_macsec_get_rx_sc_from_sc_list(const struct list_head *list, sci_t sci)
static int macsec_rx_sa_active_update(struct macsec_context *ctx, static int macsec_rx_sa_active_update(struct macsec_context *ctx,
struct mlx5e_macsec_sa *rx_sa, struct mlx5e_macsec_sa *rx_sa,
bool active) bool active, u32 *fs_id)
{ {
struct mlx5e_priv *priv = macsec_netdev_priv(ctx->netdev); struct mlx5e_priv *priv = macsec_netdev_priv(ctx->netdev);
struct mlx5e_macsec *macsec = priv->macsec; struct mlx5e_macsec *macsec = priv->macsec;
...@@ -432,11 +405,11 @@ static int macsec_rx_sa_active_update(struct macsec_context *ctx, ...@@ -432,11 +405,11 @@ static int macsec_rx_sa_active_update(struct macsec_context *ctx,
rx_sa->active = active; rx_sa->active = active;
if (!active) { if (!active) {
mlx5e_macsec_cleanup_sa(macsec, rx_sa, false); mlx5e_macsec_cleanup_sa(macsec, rx_sa, false, ctx->secy->netdev, *fs_id);
return 0; return 0;
} }
err = mlx5e_macsec_init_sa(ctx, rx_sa, true, false); err = mlx5e_macsec_init_sa(ctx, rx_sa, true, false, fs_id);
if (err) if (err)
rx_sa->active = false; rx_sa->active = false;
...@@ -558,7 +531,7 @@ static int mlx5e_macsec_add_txsa(struct macsec_context *ctx) ...@@ -558,7 +531,7 @@ static int mlx5e_macsec_add_txsa(struct macsec_context *ctx)
!tx_sa->active) !tx_sa->active)
goto out; goto out;
err = mlx5e_macsec_init_sa(ctx, tx_sa, tx_sc->encrypt, true); err = mlx5e_macsec_init_sa(ctx, tx_sa, tx_sc->encrypt, true, NULL);
if (err) if (err)
goto destroy_encryption_key; goto destroy_encryption_key;
...@@ -622,7 +595,7 @@ static int mlx5e_macsec_upd_txsa(struct macsec_context *ctx) ...@@ -622,7 +595,7 @@ static int mlx5e_macsec_upd_txsa(struct macsec_context *ctx)
goto out; goto out;
if (ctx_tx_sa->active) { if (ctx_tx_sa->active) {
err = mlx5e_macsec_init_sa(ctx, tx_sa, tx_sc->encrypt, true); err = mlx5e_macsec_init_sa(ctx, tx_sa, tx_sc->encrypt, true, NULL);
if (err) if (err)
goto out; goto out;
} else { } else {
...@@ -631,7 +604,7 @@ static int mlx5e_macsec_upd_txsa(struct macsec_context *ctx) ...@@ -631,7 +604,7 @@ static int mlx5e_macsec_upd_txsa(struct macsec_context *ctx)
goto out; goto out;
} }
mlx5e_macsec_cleanup_sa(macsec, tx_sa, true); mlx5e_macsec_cleanup_sa(macsec, tx_sa, true, ctx->secy->netdev, 0);
} }
out: out:
mutex_unlock(&macsec->lock); mutex_unlock(&macsec->lock);
...@@ -664,7 +637,7 @@ static int mlx5e_macsec_del_txsa(struct macsec_context *ctx) ...@@ -664,7 +637,7 @@ static int mlx5e_macsec_del_txsa(struct macsec_context *ctx)
goto out; goto out;
} }
mlx5e_macsec_cleanup_sa(macsec, tx_sa, true); mlx5e_macsec_cleanup_sa(macsec, tx_sa, true, ctx->secy->netdev, 0);
mlx5_destroy_encryption_key(macsec->mdev, tx_sa->enc_key_id); mlx5_destroy_encryption_key(macsec->mdev, tx_sa->enc_key_id);
kfree_rcu_mightsleep(tx_sa); kfree_rcu_mightsleep(tx_sa);
macsec_device->tx_sa[assoc_num] = NULL; macsec_device->tx_sa[assoc_num] = NULL;
...@@ -675,20 +648,6 @@ static int mlx5e_macsec_del_txsa(struct macsec_context *ctx) ...@@ -675,20 +648,6 @@ static int mlx5e_macsec_del_txsa(struct macsec_context *ctx)
return err; return err;
} }
static u32 mlx5e_macsec_get_sa_from_hashtable(struct rhashtable *sci_hash, sci_t *sci)
{
struct mlx5e_macsec_sa *macsec_sa;
u32 fs_id = 0;
rcu_read_lock();
macsec_sa = rhashtable_lookup(sci_hash, sci, rhash_sci);
if (macsec_sa)
fs_id = macsec_sa->fs_id;
rcu_read_unlock();
return fs_id;
}
static int mlx5e_macsec_add_rxsc(struct macsec_context *ctx) static int mlx5e_macsec_add_rxsc(struct macsec_context *ctx)
{ {
struct mlx5e_macsec_rx_sc_xarray_element *sc_xarray_element; struct mlx5e_macsec_rx_sc_xarray_element *sc_xarray_element;
...@@ -808,7 +767,8 @@ static int mlx5e_macsec_upd_rxsc(struct macsec_context *ctx) ...@@ -808,7 +767,8 @@ static int mlx5e_macsec_upd_rxsc(struct macsec_context *ctx)
if (!rx_sa) if (!rx_sa)
continue; continue;
err = macsec_rx_sa_active_update(ctx, rx_sa, rx_sa->active && ctx_rx_sc->active); err = macsec_rx_sa_active_update(ctx, rx_sa, rx_sa->active && ctx_rx_sc->active,
&rx_sc->sc_xarray_element->fs_id);
if (err) if (err)
goto out; goto out;
} }
...@@ -819,7 +779,8 @@ static int mlx5e_macsec_upd_rxsc(struct macsec_context *ctx) ...@@ -819,7 +779,8 @@ static int mlx5e_macsec_upd_rxsc(struct macsec_context *ctx)
return err; return err;
} }
static void macsec_del_rxsc_ctx(struct mlx5e_macsec *macsec, struct mlx5e_macsec_rx_sc *rx_sc) static void macsec_del_rxsc_ctx(struct mlx5e_macsec *macsec, struct mlx5e_macsec_rx_sc *rx_sc,
struct net_device *netdev)
{ {
struct mlx5e_macsec_sa *rx_sa; struct mlx5e_macsec_sa *rx_sa;
int i; int i;
...@@ -829,7 +790,8 @@ static void macsec_del_rxsc_ctx(struct mlx5e_macsec *macsec, struct mlx5e_macsec ...@@ -829,7 +790,8 @@ static void macsec_del_rxsc_ctx(struct mlx5e_macsec *macsec, struct mlx5e_macsec
if (!rx_sa) if (!rx_sa)
continue; continue;
mlx5e_macsec_cleanup_sa(macsec, rx_sa, false); mlx5e_macsec_cleanup_sa(macsec, rx_sa, false, netdev,
rx_sc->sc_xarray_element->fs_id);
mlx5_destroy_encryption_key(macsec->mdev, rx_sa->enc_key_id); mlx5_destroy_encryption_key(macsec->mdev, rx_sa->enc_key_id);
kfree(rx_sa); kfree(rx_sa);
...@@ -877,7 +839,7 @@ static int mlx5e_macsec_del_rxsc(struct macsec_context *ctx) ...@@ -877,7 +839,7 @@ static int mlx5e_macsec_del_rxsc(struct macsec_context *ctx)
goto out; goto out;
} }
macsec_del_rxsc_ctx(macsec, rx_sc); macsec_del_rxsc_ctx(macsec, rx_sc, ctx->secy->netdev);
out: out:
mutex_unlock(&macsec->lock); mutex_unlock(&macsec->lock);
...@@ -936,7 +898,6 @@ static int mlx5e_macsec_add_rxsa(struct macsec_context *ctx) ...@@ -936,7 +898,6 @@ static int mlx5e_macsec_add_rxsa(struct macsec_context *ctx)
rx_sa->next_pn = ctx_rx_sa->next_pn; rx_sa->next_pn = ctx_rx_sa->next_pn;
rx_sa->sci = sci; rx_sa->sci = sci;
rx_sa->assoc_num = assoc_num; rx_sa->assoc_num = assoc_num;
rx_sa->fs_id = rx_sc->sc_xarray_element->fs_id;
if (ctx->secy->xpn) if (ctx->secy->xpn)
update_macsec_epn(rx_sa, &ctx_rx_sa->key, &ctx_rx_sa->next_pn_halves, update_macsec_epn(rx_sa, &ctx_rx_sa->key, &ctx_rx_sa->next_pn_halves,
...@@ -953,7 +914,7 @@ static int mlx5e_macsec_add_rxsa(struct macsec_context *ctx) ...@@ -953,7 +914,7 @@ static int mlx5e_macsec_add_rxsa(struct macsec_context *ctx)
goto out; goto out;
//TODO - add support for both authentication and encryption flows //TODO - add support for both authentication and encryption flows
err = mlx5e_macsec_init_sa(ctx, rx_sa, true, false); err = mlx5e_macsec_init_sa(ctx, rx_sa, true, false, &rx_sc->sc_xarray_element->fs_id);
if (err) if (err)
goto destroy_encryption_key; goto destroy_encryption_key;
...@@ -1020,7 +981,8 @@ static int mlx5e_macsec_upd_rxsa(struct macsec_context *ctx) ...@@ -1020,7 +981,8 @@ static int mlx5e_macsec_upd_rxsa(struct macsec_context *ctx)
goto out; goto out;
} }
err = macsec_rx_sa_active_update(ctx, rx_sa, ctx_rx_sa->active); err = macsec_rx_sa_active_update(ctx, rx_sa, ctx_rx_sa->active,
&rx_sc->sc_xarray_element->fs_id);
out: out:
mutex_unlock(&macsec->lock); mutex_unlock(&macsec->lock);
...@@ -1068,7 +1030,8 @@ static int mlx5e_macsec_del_rxsa(struct macsec_context *ctx) ...@@ -1068,7 +1030,8 @@ static int mlx5e_macsec_del_rxsa(struct macsec_context *ctx)
goto out; goto out;
} }
mlx5e_macsec_cleanup_sa(macsec, rx_sa, false); mlx5e_macsec_cleanup_sa(macsec, rx_sa, false, ctx->secy->netdev,
rx_sc->sc_xarray_element->fs_id);
mlx5_destroy_encryption_key(macsec->mdev, rx_sa->enc_key_id); mlx5_destroy_encryption_key(macsec->mdev, rx_sa->enc_key_id);
kfree(rx_sa); kfree(rx_sa);
rx_sc->rx_sa[assoc_num] = NULL; rx_sc->rx_sa[assoc_num] = NULL;
...@@ -1149,7 +1112,8 @@ static int macsec_upd_secy_hw_address(struct macsec_context *ctx, ...@@ -1149,7 +1112,8 @@ static int macsec_upd_secy_hw_address(struct macsec_context *ctx,
if (!rx_sa || !rx_sa->macsec_rule) if (!rx_sa || !rx_sa->macsec_rule)
continue; continue;
mlx5e_macsec_cleanup_sa(macsec, rx_sa, false); mlx5e_macsec_cleanup_sa(macsec, rx_sa, false, ctx->secy->netdev,
rx_sc->sc_xarray_element->fs_id);
} }
} }
...@@ -1160,7 +1124,8 @@ static int macsec_upd_secy_hw_address(struct macsec_context *ctx, ...@@ -1160,7 +1124,8 @@ static int macsec_upd_secy_hw_address(struct macsec_context *ctx,
continue; continue;
if (rx_sa->active) { if (rx_sa->active) {
err = mlx5e_macsec_init_sa(ctx, rx_sa, true, false); err = mlx5e_macsec_init_sa(ctx, rx_sa, true, false,
&rx_sc->sc_xarray_element->fs_id);
if (err) if (err)
goto out; goto out;
} }
...@@ -1213,7 +1178,7 @@ static int mlx5e_macsec_upd_secy(struct macsec_context *ctx) ...@@ -1213,7 +1178,7 @@ static int mlx5e_macsec_upd_secy(struct macsec_context *ctx)
if (!tx_sa) if (!tx_sa)
continue; continue;
mlx5e_macsec_cleanup_sa(macsec, tx_sa, true); mlx5e_macsec_cleanup_sa(macsec, tx_sa, true, ctx->secy->netdev, 0);
} }
for (i = 0; i < MACSEC_NUM_AN; ++i) { for (i = 0; i < MACSEC_NUM_AN; ++i) {
...@@ -1222,7 +1187,7 @@ static int mlx5e_macsec_upd_secy(struct macsec_context *ctx) ...@@ -1222,7 +1187,7 @@ static int mlx5e_macsec_upd_secy(struct macsec_context *ctx)
continue; continue;
if (tx_sa->assoc_num == tx_sc->encoding_sa && tx_sa->active) { if (tx_sa->assoc_num == tx_sc->encoding_sa && tx_sa->active) {
err = mlx5e_macsec_init_sa(ctx, tx_sa, tx_sc->encrypt, true); err = mlx5e_macsec_init_sa(ctx, tx_sa, tx_sc->encrypt, true, NULL);
if (err) if (err)
goto out; goto out;
} }
...@@ -1260,7 +1225,7 @@ static int mlx5e_macsec_del_secy(struct macsec_context *ctx) ...@@ -1260,7 +1225,7 @@ static int mlx5e_macsec_del_secy(struct macsec_context *ctx)
if (!tx_sa) if (!tx_sa)
continue; continue;
mlx5e_macsec_cleanup_sa(macsec, tx_sa, true); mlx5e_macsec_cleanup_sa(macsec, tx_sa, true, ctx->secy->netdev, 0);
mlx5_destroy_encryption_key(macsec->mdev, tx_sa->enc_key_id); mlx5_destroy_encryption_key(macsec->mdev, tx_sa->enc_key_id);
kfree(tx_sa); kfree(tx_sa);
macsec_device->tx_sa[i] = NULL; macsec_device->tx_sa[i] = NULL;
...@@ -1268,7 +1233,7 @@ static int mlx5e_macsec_del_secy(struct macsec_context *ctx) ...@@ -1268,7 +1233,7 @@ static int mlx5e_macsec_del_secy(struct macsec_context *ctx)
list = &macsec_device->macsec_rx_sc_list_head; list = &macsec_device->macsec_rx_sc_list_head;
list_for_each_entry_safe(rx_sc, tmp, list, rx_sc_list_element) list_for_each_entry_safe(rx_sc, tmp, list, rx_sc_list_element)
macsec_del_rxsc_ctx(macsec, rx_sc); macsec_del_rxsc_ctx(macsec, rx_sc, ctx->secy->netdev);
kfree(macsec_device->dev_addr); kfree(macsec_device->dev_addr);
macsec_device->dev_addr = NULL; macsec_device->dev_addr = NULL;
...@@ -1693,7 +1658,8 @@ bool mlx5e_macsec_handle_tx_skb(struct mlx5e_macsec *macsec, struct sk_buff *skb ...@@ -1693,7 +1658,8 @@ bool mlx5e_macsec_handle_tx_skb(struct mlx5e_macsec *macsec, struct sk_buff *skb
struct metadata_dst *md_dst = skb_metadata_dst(skb); struct metadata_dst *md_dst = skb_metadata_dst(skb);
u32 fs_id; u32 fs_id;
fs_id = mlx5e_macsec_get_sa_from_hashtable(&macsec->sci_hash, &md_dst->u.macsec_info.sci); fs_id = mlx5_macsec_fs_get_fs_id_from_hashtable(macsec->mdev->macsec_fs,
&md_dst->u.macsec_info.sci);
if (!fs_id) if (!fs_id)
goto err_out; goto err_out;
...@@ -1711,7 +1677,8 @@ void mlx5e_macsec_tx_build_eseg(struct mlx5e_macsec *macsec, ...@@ -1711,7 +1677,8 @@ void mlx5e_macsec_tx_build_eseg(struct mlx5e_macsec *macsec,
struct metadata_dst *md_dst = skb_metadata_dst(skb); struct metadata_dst *md_dst = skb_metadata_dst(skb);
u32 fs_id; u32 fs_id;
fs_id = mlx5e_macsec_get_sa_from_hashtable(&macsec->sci_hash, &md_dst->u.macsec_info.sci); fs_id = mlx5_macsec_fs_get_fs_id_from_hashtable(macsec->mdev->macsec_fs,
&md_dst->u.macsec_info.sci);
if (!fs_id) if (!fs_id)
return; return;
...@@ -1779,13 +1746,6 @@ int mlx5e_macsec_init(struct mlx5e_priv *priv) ...@@ -1779,13 +1746,6 @@ int mlx5e_macsec_init(struct mlx5e_priv *priv)
INIT_LIST_HEAD(&macsec->macsec_device_list_head); INIT_LIST_HEAD(&macsec->macsec_device_list_head);
mutex_init(&macsec->lock); mutex_init(&macsec->lock);
err = rhashtable_init(&macsec->sci_hash, &rhash_sci);
if (err) {
mlx5_core_err(mdev, "MACsec offload: Failed to init SCI hash table, err=%d\n",
err);
goto err_hash;
}
err = mlx5e_macsec_aso_init(&macsec->aso, priv->mdev); err = mlx5e_macsec_aso_init(&macsec->aso, priv->mdev);
if (err) { if (err) {
mlx5_core_err(mdev, "MACsec offload: Failed to init aso, err=%d\n", err); mlx5_core_err(mdev, "MACsec offload: Failed to init aso, err=%d\n", err);
...@@ -1824,8 +1784,6 @@ int mlx5e_macsec_init(struct mlx5e_priv *priv) ...@@ -1824,8 +1784,6 @@ int mlx5e_macsec_init(struct mlx5e_priv *priv)
err_wq: err_wq:
mlx5e_macsec_aso_cleanup(&macsec->aso, priv->mdev); mlx5e_macsec_aso_cleanup(&macsec->aso, priv->mdev);
err_aso: err_aso:
rhashtable_destroy(&macsec->sci_hash);
err_hash:
kfree(macsec); kfree(macsec);
priv->macsec = NULL; priv->macsec = NULL;
return err; return err;
...@@ -1843,7 +1801,6 @@ void mlx5e_macsec_cleanup(struct mlx5e_priv *priv) ...@@ -1843,7 +1801,6 @@ void mlx5e_macsec_cleanup(struct mlx5e_priv *priv)
mlx5_macsec_fs_cleanup(mdev->macsec_fs); mlx5_macsec_fs_cleanup(mdev->macsec_fs);
destroy_workqueue(macsec->wq); destroy_workqueue(macsec->wq);
mlx5e_macsec_aso_cleanup(&macsec->aso, mdev); mlx5e_macsec_aso_cleanup(&macsec->aso, mdev);
rhashtable_destroy(&macsec->sci_hash);
mutex_destroy(&macsec->lock); mutex_destroy(&macsec->lock);
kfree(macsec); kfree(macsec);
} }
...@@ -74,6 +74,20 @@ struct mlx5_macsec_tables { ...@@ -74,6 +74,20 @@ struct mlx5_macsec_tables {
u32 refcnt; u32 refcnt;
}; };
struct mlx5_fs_id {
u32 id;
refcount_t refcnt;
sci_t sci;
struct rhash_head hash;
};
struct mlx5_macsec_device {
struct list_head macsec_devices_list_entry;
void *macdev;
struct xarray tx_id_xa;
struct xarray rx_id_xa;
};
struct mlx5_macsec_tx { struct mlx5_macsec_tx {
struct mlx5_flow_handle *crypto_mke_rule; struct mlx5_flow_handle *crypto_mke_rule;
struct mlx5_flow_handle *check_rule; struct mlx5_flow_handle *check_rule;
...@@ -100,6 +114,22 @@ union mlx5_macsec_rule { ...@@ -100,6 +114,22 @@ union mlx5_macsec_rule {
struct mlx5_macsec_rx_rule rx_rule; struct mlx5_macsec_rx_rule rx_rule;
}; };
static const struct rhashtable_params rhash_sci = {
.key_len = sizeof_field(struct mlx5_fs_id, sci),
.key_offset = offsetof(struct mlx5_fs_id, sci),
.head_offset = offsetof(struct mlx5_fs_id, hash),
.automatic_shrinking = true,
.min_size = 1,
};
static const struct rhashtable_params rhash_fs_id = {
.key_len = sizeof_field(struct mlx5_fs_id, id),
.key_offset = offsetof(struct mlx5_fs_id, id),
.head_offset = offsetof(struct mlx5_fs_id, hash),
.automatic_shrinking = true,
.min_size = 1,
};
struct mlx5_macsec_fs { struct mlx5_macsec_fs {
struct mlx5_core_dev *mdev; struct mlx5_core_dev *mdev;
struct mlx5_macsec_tx *tx_fs; struct mlx5_macsec_tx *tx_fs;
...@@ -107,6 +137,15 @@ struct mlx5_macsec_fs { ...@@ -107,6 +137,15 @@ struct mlx5_macsec_fs {
/* Stats manage */ /* Stats manage */
struct mlx5_macsec_stats stats; struct mlx5_macsec_stats stats;
/* Tx sci -> fs id mapping handling */
struct rhashtable sci_hash; /* sci -> mlx5_fs_id */
/* RX fs_id -> mlx5_fs_id mapping handling */
struct rhashtable fs_id_hash; /* fs_id -> mlx5_fs_id */
/* TX & RX fs_id lists per macsec device */
struct list_head macsec_devices_list;
}; };
static void macsec_fs_destroy_groups(struct mlx5_macsec_flow_table *ft) static void macsec_fs_destroy_groups(struct mlx5_macsec_flow_table *ft)
...@@ -512,9 +551,137 @@ static void macsec_fs_tx_create_sectag_header(const struct macsec_context *ctx, ...@@ -512,9 +551,137 @@ static void macsec_fs_tx_create_sectag_header(const struct macsec_context *ctx,
memcpy(reformatbf, &sectag, *reformat_size); memcpy(reformatbf, &sectag, *reformat_size);
} }
static bool macsec_fs_is_macsec_device_empty(struct mlx5_macsec_device *macsec_device)
{
if (xa_empty(&macsec_device->tx_id_xa) &&
xa_empty(&macsec_device->rx_id_xa))
return true;
return false;
}
static void macsec_fs_id_del(struct list_head *macsec_devices_list, u32 fs_id,
void *macdev, struct rhashtable *hash_table, bool is_tx)
{
const struct rhashtable_params *rhash = (is_tx) ? &rhash_sci : &rhash_fs_id;
struct mlx5_macsec_device *iter, *macsec_device = NULL;
struct mlx5_fs_id *fs_id_found;
struct xarray *fs_id_xa;
list_for_each_entry(iter, macsec_devices_list, macsec_devices_list_entry) {
if (iter->macdev == macdev) {
macsec_device = iter;
break;
}
}
WARN_ON(!macsec_device);
fs_id_xa = (is_tx) ? &macsec_device->tx_id_xa :
&macsec_device->rx_id_xa;
xa_lock(fs_id_xa);
fs_id_found = xa_load(fs_id_xa, fs_id);
WARN_ON(!fs_id_found);
if (!refcount_dec_and_test(&fs_id_found->refcnt)) {
xa_unlock(fs_id_xa);
return;
}
if (fs_id_found->id) {
/* Make sure ongoing datapath readers sees a valid SA */
rhashtable_remove_fast(hash_table, &fs_id_found->hash, *rhash);
fs_id_found->id = 0;
}
xa_unlock(fs_id_xa);
xa_erase(fs_id_xa, fs_id);
kfree(fs_id_found);
if (macsec_fs_is_macsec_device_empty(macsec_device)) {
list_del(&macsec_device->macsec_devices_list_entry);
kfree(macsec_device);
}
}
static int macsec_fs_id_add(struct list_head *macsec_devices_list, u32 fs_id,
void *macdev, struct rhashtable *hash_table, sci_t sci,
bool is_tx)
{
const struct rhashtable_params *rhash = (is_tx) ? &rhash_sci : &rhash_fs_id;
struct mlx5_macsec_device *iter, *macsec_device = NULL;
struct mlx5_fs_id *fs_id_iter;
struct xarray *fs_id_xa;
int err;
if (!is_tx) {
rcu_read_lock();
fs_id_iter = rhashtable_lookup(hash_table, &fs_id, rhash_fs_id);
if (fs_id_iter) {
refcount_inc(&fs_id_iter->refcnt);
rcu_read_unlock();
return 0;
}
rcu_read_unlock();
}
fs_id_iter = kzalloc(sizeof(*fs_id_iter), GFP_KERNEL);
if (!fs_id_iter)
return -ENOMEM;
list_for_each_entry(iter, macsec_devices_list, macsec_devices_list_entry) {
if (iter->macdev == macdev) {
macsec_device = iter;
break;
}
}
if (!macsec_device) { /* first time adding a SA to that device */
macsec_device = kzalloc(sizeof(*macsec_device), GFP_KERNEL);
if (!macsec_device) {
err = -ENOMEM;
goto err_alloc_dev;
}
macsec_device->macdev = macdev;
xa_init(&macsec_device->tx_id_xa);
xa_init(&macsec_device->rx_id_xa);
list_add(&macsec_device->macsec_devices_list_entry, macsec_devices_list);
}
fs_id_xa = (is_tx) ? &macsec_device->tx_id_xa :
&macsec_device->rx_id_xa;
fs_id_iter->id = fs_id;
refcount_set(&fs_id_iter->refcnt, 1);
fs_id_iter->sci = sci;
err = xa_err(xa_store(fs_id_xa, fs_id, fs_id_iter, GFP_KERNEL));
if (err)
goto err_store_id;
err = rhashtable_insert_fast(hash_table, &fs_id_iter->hash, *rhash);
if (err)
goto err_hash_insert;
return 0;
err_hash_insert:
xa_erase(fs_id_xa, fs_id);
err_store_id:
if (macsec_fs_is_macsec_device_empty(macsec_device)) {
list_del(&macsec_device->macsec_devices_list_entry);
kfree(macsec_device);
}
err_alloc_dev:
kfree(fs_id_iter);
return err;
}
static void macsec_fs_tx_del_rule(struct mlx5_macsec_fs *macsec_fs, static void macsec_fs_tx_del_rule(struct mlx5_macsec_fs *macsec_fs,
struct mlx5_macsec_tx_rule *tx_rule) struct mlx5_macsec_tx_rule *tx_rule,
void *macdev)
{ {
macsec_fs_id_del(&macsec_fs->macsec_devices_list, tx_rule->fs_id, macdev,
&macsec_fs->sci_hash, true);
if (tx_rule->rule) { if (tx_rule->rule) {
mlx5_del_flow_rules(tx_rule->rule); mlx5_del_flow_rules(tx_rule->rule);
tx_rule->rule = NULL; tx_rule->rule = NULL;
...@@ -540,8 +707,7 @@ static void macsec_fs_tx_del_rule(struct mlx5_macsec_fs *macsec_fs, ...@@ -540,8 +707,7 @@ static void macsec_fs_tx_del_rule(struct mlx5_macsec_fs *macsec_fs,
static union mlx5_macsec_rule * static union mlx5_macsec_rule *
macsec_fs_tx_add_rule(struct mlx5_macsec_fs *macsec_fs, macsec_fs_tx_add_rule(struct mlx5_macsec_fs *macsec_fs,
const struct macsec_context *macsec_ctx, const struct macsec_context *macsec_ctx,
struct mlx5_macsec_rule_attrs *attrs, struct mlx5_macsec_rule_attrs *attrs)
u32 *sa_fs_id)
{ {
char reformatbf[MLX5_MACSEC_TAG_LEN + MACSEC_SCI_LEN]; char reformatbf[MLX5_MACSEC_TAG_LEN + MACSEC_SCI_LEN];
struct mlx5_pkt_reformat_params reformat_params = {}; struct mlx5_pkt_reformat_params reformat_params = {};
...@@ -605,7 +771,6 @@ macsec_fs_tx_add_rule(struct mlx5_macsec_fs *macsec_fs, ...@@ -605,7 +771,6 @@ macsec_fs_tx_add_rule(struct mlx5_macsec_fs *macsec_fs,
} }
tx_rule->fs_id = fs_id; tx_rule->fs_id = fs_id;
*sa_fs_id = fs_id;
flow_act.action = MLX5_FLOW_CONTEXT_ACTION_FWD_DEST | flow_act.action = MLX5_FLOW_CONTEXT_ACTION_FWD_DEST |
MLX5_FLOW_CONTEXT_ACTION_CRYPTO_ENCRYPT | MLX5_FLOW_CONTEXT_ACTION_CRYPTO_ENCRYPT |
...@@ -620,10 +785,17 @@ macsec_fs_tx_add_rule(struct mlx5_macsec_fs *macsec_fs, ...@@ -620,10 +785,17 @@ macsec_fs_tx_add_rule(struct mlx5_macsec_fs *macsec_fs,
} }
tx_rule->rule = rule; tx_rule->rule = rule;
err = macsec_fs_id_add(&macsec_fs->macsec_devices_list, fs_id, macsec_ctx->secy->netdev,
&macsec_fs->sci_hash, attrs->sci, true);
if (err) {
mlx5_core_err(mdev, "Failed to save fs_id, err=%d\n", err);
goto err;
}
goto out_spec; goto out_spec;
err: err:
macsec_fs_tx_del_rule(macsec_fs, tx_rule); macsec_fs_tx_del_rule(macsec_fs, tx_rule, macsec_ctx->secy->netdev);
macsec_rule = NULL; macsec_rule = NULL;
out_spec: out_spec:
kvfree(spec); kvfree(spec);
...@@ -699,6 +871,7 @@ static int macsec_fs_tx_init(struct mlx5_macsec_fs *macsec_fs) ...@@ -699,6 +871,7 @@ static int macsec_fs_tx_init(struct mlx5_macsec_fs *macsec_fs)
tx_tables->check_miss_rule_counter = flow_counter; tx_tables->check_miss_rule_counter = flow_counter;
ida_init(&tx_fs->tx_halloc); ida_init(&tx_fs->tx_halloc);
INIT_LIST_HEAD(&macsec_fs->macsec_devices_list);
macsec_fs->tx_fs = tx_fs; macsec_fs->tx_fs = tx_fs;
...@@ -1070,10 +1243,14 @@ static void macsec_fs_rx_ft_put(struct mlx5_macsec_fs *macsec_fs) ...@@ -1070,10 +1243,14 @@ static void macsec_fs_rx_ft_put(struct mlx5_macsec_fs *macsec_fs)
} }
static void macsec_fs_rx_del_rule(struct mlx5_macsec_fs *macsec_fs, static void macsec_fs_rx_del_rule(struct mlx5_macsec_fs *macsec_fs,
struct mlx5_macsec_rx_rule *rx_rule) struct mlx5_macsec_rx_rule *rx_rule,
void *macdev, u32 fs_id)
{ {
int i; int i;
macsec_fs_id_del(&macsec_fs->macsec_devices_list, fs_id, macdev,
&macsec_fs->fs_id_hash, false);
for (i = 0; i < RX_NUM_OF_RULES_PER_SA; ++i) { for (i = 0; i < RX_NUM_OF_RULES_PER_SA; ++i) {
if (rx_rule->rule[i]) { if (rx_rule->rule[i]) {
mlx5_del_flow_rules(rx_rule->rule[i]); mlx5_del_flow_rules(rx_rule->rule[i]);
...@@ -1139,6 +1316,7 @@ static void macsec_fs_rx_setup_fte(struct mlx5_flow_spec *spec, ...@@ -1139,6 +1316,7 @@ static void macsec_fs_rx_setup_fte(struct mlx5_flow_spec *spec,
static union mlx5_macsec_rule * static union mlx5_macsec_rule *
macsec_fs_rx_add_rule(struct mlx5_macsec_fs *macsec_fs, macsec_fs_rx_add_rule(struct mlx5_macsec_fs *macsec_fs,
const struct macsec_context *macsec_ctx,
struct mlx5_macsec_rule_attrs *attrs, struct mlx5_macsec_rule_attrs *attrs,
u32 fs_id) u32 fs_id)
{ {
...@@ -1238,11 +1416,18 @@ macsec_fs_rx_add_rule(struct mlx5_macsec_fs *macsec_fs, ...@@ -1238,11 +1416,18 @@ macsec_fs_rx_add_rule(struct mlx5_macsec_fs *macsec_fs,
rx_rule->rule[1] = rule; rx_rule->rule[1] = rule;
} }
err = macsec_fs_id_add(&macsec_fs->macsec_devices_list, fs_id, macsec_ctx->secy->netdev,
&macsec_fs->fs_id_hash, attrs->sci, false);
if (err) {
mlx5_core_err(mdev, "Failed to save fs_id, err=%d\n", err);
goto err;
}
kvfree(spec); kvfree(spec);
return macsec_rule; return macsec_rule;
err: err:
macsec_fs_rx_del_rule(macsec_fs, rx_rule); macsec_fs_rx_del_rule(macsec_fs, rx_rule, macsec_ctx->secy->netdev, fs_id);
macsec_rule = NULL; macsec_rule = NULL;
out_spec: out_spec:
kvfree(spec); kvfree(spec);
...@@ -1362,6 +1547,20 @@ struct mlx5_macsec_stats *mlx5_macsec_fs_get_stats(struct mlx5_macsec_fs *macsec ...@@ -1362,6 +1547,20 @@ struct mlx5_macsec_stats *mlx5_macsec_fs_get_stats(struct mlx5_macsec_fs *macsec
return &macsec_fs->stats; return &macsec_fs->stats;
} }
u32 mlx5_macsec_fs_get_fs_id_from_hashtable(struct mlx5_macsec_fs *macsec_fs, sci_t *sci)
{
struct mlx5_fs_id *mlx5_fs_id;
u32 fs_id = 0;
rcu_read_lock();
mlx5_fs_id = rhashtable_lookup(&macsec_fs->sci_hash, sci, rhash_sci);
if (mlx5_fs_id)
fs_id = mlx5_fs_id->id;
rcu_read_unlock();
return fs_id;
}
union mlx5_macsec_rule * union mlx5_macsec_rule *
mlx5_macsec_fs_add_rule(struct mlx5_macsec_fs *macsec_fs, mlx5_macsec_fs_add_rule(struct mlx5_macsec_fs *macsec_fs,
const struct macsec_context *macsec_ctx, const struct macsec_context *macsec_ctx,
...@@ -1369,23 +1568,25 @@ mlx5_macsec_fs_add_rule(struct mlx5_macsec_fs *macsec_fs, ...@@ -1369,23 +1568,25 @@ mlx5_macsec_fs_add_rule(struct mlx5_macsec_fs *macsec_fs,
u32 *sa_fs_id) u32 *sa_fs_id)
{ {
return (attrs->action == MLX5_ACCEL_MACSEC_ACTION_ENCRYPT) ? return (attrs->action == MLX5_ACCEL_MACSEC_ACTION_ENCRYPT) ?
macsec_fs_tx_add_rule(macsec_fs, macsec_ctx, attrs, sa_fs_id) : macsec_fs_tx_add_rule(macsec_fs, macsec_ctx, attrs) :
macsec_fs_rx_add_rule(macsec_fs, attrs, *sa_fs_id); macsec_fs_rx_add_rule(macsec_fs, macsec_ctx, attrs, *sa_fs_id);
} }
void mlx5_macsec_fs_del_rule(struct mlx5_macsec_fs *macsec_fs, void mlx5_macsec_fs_del_rule(struct mlx5_macsec_fs *macsec_fs,
union mlx5_macsec_rule *macsec_rule, union mlx5_macsec_rule *macsec_rule,
int action) int action, void *macdev, u32 sa_fs_id)
{ {
(action == MLX5_ACCEL_MACSEC_ACTION_ENCRYPT) ? (action == MLX5_ACCEL_MACSEC_ACTION_ENCRYPT) ?
macsec_fs_tx_del_rule(macsec_fs, &macsec_rule->tx_rule) : macsec_fs_tx_del_rule(macsec_fs, &macsec_rule->tx_rule, macdev) :
macsec_fs_rx_del_rule(macsec_fs, &macsec_rule->rx_rule); macsec_fs_rx_del_rule(macsec_fs, &macsec_rule->rx_rule, macdev, sa_fs_id);
} }
void mlx5_macsec_fs_cleanup(struct mlx5_macsec_fs *macsec_fs) void mlx5_macsec_fs_cleanup(struct mlx5_macsec_fs *macsec_fs)
{ {
macsec_fs_rx_cleanup(macsec_fs); macsec_fs_rx_cleanup(macsec_fs);
macsec_fs_tx_cleanup(macsec_fs); macsec_fs_tx_cleanup(macsec_fs);
rhashtable_destroy(&macsec_fs->fs_id_hash);
rhashtable_destroy(&macsec_fs->sci_hash);
kfree(macsec_fs); kfree(macsec_fs);
} }
...@@ -1401,10 +1602,24 @@ mlx5_macsec_fs_init(struct mlx5_core_dev *mdev) ...@@ -1401,10 +1602,24 @@ mlx5_macsec_fs_init(struct mlx5_core_dev *mdev)
macsec_fs->mdev = mdev; macsec_fs->mdev = mdev;
err = rhashtable_init(&macsec_fs->sci_hash, &rhash_sci);
if (err) {
mlx5_core_err(mdev, "MACsec offload: Failed to init SCI hash table, err=%d\n",
err);
goto err_hash;
}
err = rhashtable_init(&macsec_fs->fs_id_hash, &rhash_fs_id);
if (err) {
mlx5_core_err(mdev, "MACsec offload: Failed to init FS_ID hash table, err=%d\n",
err);
goto sci_hash_cleanup;
}
err = macsec_fs_tx_init(macsec_fs); err = macsec_fs_tx_init(macsec_fs);
if (err) { if (err) {
mlx5_core_err(mdev, "MACsec offload: Failed to init tx_fs, err=%d\n", err); mlx5_core_err(mdev, "MACsec offload: Failed to init tx_fs, err=%d\n", err);
goto err; goto fs_id_hash_cleanup;
} }
err = macsec_fs_rx_init(macsec_fs); err = macsec_fs_rx_init(macsec_fs);
...@@ -1417,7 +1632,11 @@ mlx5_macsec_fs_init(struct mlx5_core_dev *mdev) ...@@ -1417,7 +1632,11 @@ mlx5_macsec_fs_init(struct mlx5_core_dev *mdev)
tx_cleanup: tx_cleanup:
macsec_fs_tx_cleanup(macsec_fs); macsec_fs_tx_cleanup(macsec_fs);
err: fs_id_hash_cleanup:
rhashtable_destroy(&macsec_fs->fs_id_hash);
sci_hash_cleanup:
rhashtable_destroy(&macsec_fs->sci_hash);
err_hash:
kfree(macsec_fs); kfree(macsec_fs);
return NULL; return NULL;
} }
...@@ -53,10 +53,11 @@ mlx5_macsec_fs_add_rule(struct mlx5_macsec_fs *macsec_fs, ...@@ -53,10 +53,11 @@ mlx5_macsec_fs_add_rule(struct mlx5_macsec_fs *macsec_fs,
void mlx5_macsec_fs_del_rule(struct mlx5_macsec_fs *macsec_fs, void mlx5_macsec_fs_del_rule(struct mlx5_macsec_fs *macsec_fs,
union mlx5_macsec_rule *macsec_rule, union mlx5_macsec_rule *macsec_rule,
int action); int action, void *macdev, u32 sa_fs_id);
void mlx5_macsec_fs_get_stats_fill(struct mlx5_macsec_fs *macsec_fs, void *macsec_stats); void mlx5_macsec_fs_get_stats_fill(struct mlx5_macsec_fs *macsec_fs, void *macsec_stats);
struct mlx5_macsec_stats *mlx5_macsec_fs_get_stats(struct mlx5_macsec_fs *macsec_fs); struct mlx5_macsec_stats *mlx5_macsec_fs_get_stats(struct mlx5_macsec_fs *macsec_fs);
u32 mlx5_macsec_fs_get_fs_id_from_hashtable(struct mlx5_macsec_fs *macsec_fs, sci_t *sci);
#endif #endif
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment