/* * Copyright (c) 2019, Mellanox Technologies. All rights reserved. * * This software is available to you under a choice of one of two * licenses. You may choose to be licensed under the terms of the GNU * General Public License (GPL) Version 2, available from the file * COPYING in the main directory of this source tree, or the * OpenIB.org BSD license below: * * Redistribution and use in source and binary forms, with or * without modification, are permitted provided that the following * conditions are met: * * - Redistributions of source code must retain the above * copyright notice, this list of conditions and the following * disclaimer. * * - Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials * provided with the distribution. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #include #include "mlx5dv_dr.h" static void dr_table_uninit_nic(struct dr_table_rx_tx *nic_tbl) { dr_htbl_put(nic_tbl->s_anchor); } static void dr_table_uninit_fdb(struct mlx5dv_dr_table *tbl) { dr_table_uninit_nic(&tbl->rx); dr_table_uninit_nic(&tbl->tx); } static void dr_table_uninit(struct mlx5dv_dr_table *tbl) { switch (tbl->dmn->type) { case MLX5DV_DR_DOMAIN_TYPE_NIC_RX: dr_table_uninit_nic(&tbl->rx); break; case MLX5DV_DR_DOMAIN_TYPE_NIC_TX: dr_table_uninit_nic(&tbl->tx); break; case MLX5DV_DR_DOMAIN_TYPE_FDB: dr_table_uninit_fdb(tbl); break; default: break; } } static int dr_table_init_nic(struct mlx5dv_dr_domain *dmn, struct dr_table_rx_tx *nic_tbl) { struct dr_domain_rx_tx *nic_dmn = nic_tbl->nic_dmn; struct dr_htbl_connect_info info; int ret; nic_tbl->s_anchor = dr_ste_htbl_alloc(dmn->ste_icm_pool, DR_CHUNK_SIZE_1, DR_STE_HTBL_TYPE_LEGACY, DR_STE_LU_TYPE_DONT_CARE, 0); if (!nic_tbl->s_anchor) return errno; info.type = CONNECT_MISS; info.miss_icm_addr = nic_dmn->default_icm_addr; ret = dr_ste_htbl_init_and_postsend(dmn, nic_dmn, nic_tbl->s_anchor, &info, true, 0); if (ret) goto free_s_anchor; dr_htbl_get(nic_tbl->s_anchor); return 0; free_s_anchor: dr_ste_htbl_free(nic_tbl->s_anchor); return ret; } static int dr_table_init_fdb(struct mlx5dv_dr_table *tbl) { int ret; ret = dr_table_init_nic(tbl->dmn, &tbl->rx); if (ret) return ret; ret = dr_table_init_nic(tbl->dmn, &tbl->tx); if (ret) goto destroy_rx; return 0; destroy_rx: dr_table_uninit_nic(&tbl->rx); return ret; } static int dr_table_init(struct mlx5dv_dr_table *tbl) { int ret = 0; list_head_init(&tbl->matcher_list); switch (tbl->dmn->type) { case MLX5DV_DR_DOMAIN_TYPE_NIC_RX: tbl->table_type = FS_FT_NIC_RX; tbl->rx.nic_dmn = &tbl->dmn->info.rx; ret = dr_table_init_nic(tbl->dmn, &tbl->rx); break; case MLX5DV_DR_DOMAIN_TYPE_NIC_TX: tbl->table_type = FS_FT_NIC_TX; tbl->tx.nic_dmn = &tbl->dmn->info.tx; ret = dr_table_init_nic(tbl->dmn, &tbl->tx); break; case MLX5DV_DR_DOMAIN_TYPE_FDB: tbl->table_type = FS_FT_FDB; tbl->rx.nic_dmn = &tbl->dmn->info.rx; tbl->tx.nic_dmn = &tbl->dmn->info.tx; ret = dr_table_init_fdb(tbl); break; default: assert(false); break; } return ret; } static int dr_table_create_devx_tbl(struct mlx5dv_dr_table *tbl) { struct dr_devx_flow_table_attr ft_attr = {}; ft_attr.type = tbl->table_type; ft_attr.level = tbl->dmn->info.caps.max_ft_level - 1; ft_attr.sw_owner = true; if (tbl->rx.s_anchor) ft_attr.icm_addr_rx = dr_icm_pool_get_chunk_icm_addr(tbl->rx.s_anchor->chunk); if (tbl->tx.s_anchor) ft_attr.icm_addr_tx = dr_icm_pool_get_chunk_icm_addr(tbl->tx.s_anchor->chunk); tbl->devx_obj = dr_devx_create_flow_table(tbl->dmn->ctx, &ft_attr); if (!tbl->devx_obj) return errno; return 0; } struct mlx5dv_dr_table *mlx5dv_dr_table_create(struct mlx5dv_dr_domain *dmn, uint32_t level) { struct mlx5dv_dr_table *tbl; int ret; atomic_fetch_add(&dmn->refcount, 1); if (level && !dmn->info.supp_sw_steering) { errno = EOPNOTSUPP; goto dec_ref; } tbl = calloc(1, sizeof(*tbl)); if (!tbl) { errno = ENOMEM; goto dec_ref; } tbl->dmn = dmn; tbl->level = level; atomic_init(&tbl->refcount, 1); if (!dr_is_root_table(tbl)) { ret = dr_table_init(tbl); if (ret) goto free_tbl; ret = dr_send_ring_force_drain(dmn); if (ret) goto uninit_tbl; ret = dr_table_create_devx_tbl(tbl); if (ret) goto uninit_tbl; } list_node_init(&tbl->tbl_list); dr_domain_lock(dmn); list_add_tail(&dmn->tbl_list, &tbl->tbl_list); dr_domain_unlock(dmn); return tbl; uninit_tbl: dr_table_uninit(tbl); free_tbl: free(tbl); dec_ref: atomic_fetch_sub(&dmn->refcount, 1); return NULL; } int mlx5dv_dr_table_destroy(struct mlx5dv_dr_table *tbl) { int ret = 0; if (atomic_load(&tbl->refcount) > 1) return EBUSY; if (!dr_is_root_table(tbl)) { ret = mlx5dv_devx_obj_destroy(tbl->devx_obj); if (ret) return ret; } dr_domain_lock(tbl->dmn); list_del(&tbl->tbl_list); dr_domain_unlock(tbl->dmn); if (!dr_is_root_table(tbl)) dr_table_uninit(tbl); atomic_fetch_sub(&tbl->dmn->refcount, 1); free(tbl); return ret; }