/*- * SPDX-License-Identifier: GPL-2.0 or Linux-OpenIB * * Copyright (c) 2017 - 2023 Intel Corporation * * 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 * OpenFabrics.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 "osdep.h" #include "irdma_hmc.h" #include "irdma_defs.h" #include "irdma_type.h" #include "irdma_protos.h" #include "irdma_ws.h" /** * irdma_alloc_node - Allocate a WS node and init * @vsi: vsi pointer * @user_pri: user priority * @node_type: Type of node, leaf or parent * @parent: parent node pointer */ static struct irdma_ws_node * irdma_alloc_node(struct irdma_sc_vsi *vsi, u8 user_pri, enum irdma_ws_node_type node_type, struct irdma_ws_node *parent) { struct irdma_virt_mem ws_mem; struct irdma_ws_node *node; u16 node_index = 0; ws_mem.size = sizeof(struct irdma_ws_node); ws_mem.va = kzalloc(ws_mem.size, GFP_KERNEL); if (!ws_mem.va) return NULL; if (parent) { node_index = irdma_alloc_ws_node_id(vsi->dev); if (node_index == IRDMA_WS_NODE_INVALID) { kfree(ws_mem.va); return NULL; } } node = ws_mem.va; node->index = node_index; node->vsi_index = vsi->vsi_idx; INIT_LIST_HEAD(&node->child_list_head); if (node_type == WS_NODE_TYPE_LEAF) { node->type_leaf = true; node->traffic_class = vsi->qos[user_pri].traffic_class; node->user_pri = user_pri; node->rel_bw = vsi->qos[user_pri].rel_bw; if (!node->rel_bw) node->rel_bw = 1; node->prio_type = IRDMA_PRIO_WEIGHTED_RR; } else { node->rel_bw = 1; node->prio_type = IRDMA_PRIO_WEIGHTED_RR; node->enable = true; } node->parent = parent; return node; } /** * irdma_free_node - Free a WS node * @vsi: VSI stricture of device * @node: Pointer to node to free */ static void irdma_free_node(struct irdma_sc_vsi *vsi, struct irdma_ws_node *node) { struct irdma_virt_mem ws_mem; if (node->index) irdma_free_ws_node_id(vsi->dev, node->index); ws_mem.va = node; ws_mem.size = sizeof(struct irdma_ws_node); kfree(ws_mem.va); } /** * irdma_ws_cqp_cmd - Post CQP work scheduler node cmd * @vsi: vsi pointer * @node: pointer to node * @cmd: add, remove or modify */ static int irdma_ws_cqp_cmd(struct irdma_sc_vsi *vsi, struct irdma_ws_node *node, u8 cmd) { struct irdma_ws_node_info node_info = {0}; node_info.id = node->index; node_info.vsi = node->vsi_index; if (node->parent) node_info.parent_id = node->parent->index; else node_info.parent_id = node_info.id; node_info.weight = node->rel_bw; node_info.tc = node->traffic_class; node_info.prio_type = node->prio_type; node_info.type_leaf = node->type_leaf; node_info.enable = node->enable; if (irdma_cqp_ws_node_cmd(vsi->dev, cmd, &node_info)) { irdma_debug(vsi->dev, IRDMA_DEBUG_WS, "CQP WS CMD failed\n"); return -ENOMEM; } if (node->type_leaf && cmd == IRDMA_OP_WS_ADD_NODE) { node->qs_handle = node_info.qs_handle; vsi->qos[node->user_pri].qs_handle = node_info.qs_handle; } return 0; } /** * ws_find_node - Find SC WS node based on VSI id or TC * @parent: parent node of First VSI or TC node * @match_val: value to match * @type: match type VSI/TC */ static struct irdma_ws_node * ws_find_node(struct irdma_ws_node *parent, u16 match_val, enum irdma_ws_match_type type) { struct irdma_ws_node *node; switch (type) { case WS_MATCH_TYPE_VSI: list_for_each_entry(node, &parent->child_list_head, siblings) { if (node->vsi_index == match_val) return node; } break; case WS_MATCH_TYPE_TC: list_for_each_entry(node, &parent->child_list_head, siblings) { if (node->traffic_class == match_val) return node; } break; default: break; } return NULL; } /** * irdma_ws_in_use - Checks to see if a leaf node is in use * @vsi: vsi pointer * @user_pri: user priority */ static bool irdma_ws_in_use(struct irdma_sc_vsi *vsi, u8 user_pri) { int i; mutex_lock(&vsi->qos[user_pri].qos_mutex); if (!list_empty(&vsi->qos[user_pri].qplist)) { mutex_unlock(&vsi->qos[user_pri].qos_mutex); return true; } /* * Check if the qs handle associated with the given user priority is in use by any other user priority. If so, * nothing left to do */ for (i = 0; i < IRDMA_MAX_USER_PRIORITY; i++) { if (vsi->qos[i].qs_handle == vsi->qos[user_pri].qs_handle && !list_empty(&vsi->qos[i].qplist)) { mutex_unlock(&vsi->qos[user_pri].qos_mutex); return true; } } mutex_unlock(&vsi->qos[user_pri].qos_mutex); return false; } /** * irdma_remove_leaf - Remove leaf node unconditionally * @vsi: vsi pointer * @user_pri: user priority */ static void irdma_remove_leaf(struct irdma_sc_vsi *vsi, u8 user_pri) { struct irdma_ws_node *ws_tree_root, *vsi_node, *tc_node; u16 qs_handle; int i; qs_handle = vsi->qos[user_pri].qs_handle; for (i = 0; i < IRDMA_MAX_USER_PRIORITY; i++) if (vsi->qos[i].qs_handle == qs_handle) vsi->qos[i].valid = false; ws_tree_root = vsi->dev->ws_tree_root; if (!ws_tree_root) return; vsi_node = ws_find_node(ws_tree_root, vsi->vsi_idx, WS_MATCH_TYPE_VSI); if (!vsi_node) return; tc_node = ws_find_node(vsi_node, vsi->qos[user_pri].traffic_class, WS_MATCH_TYPE_TC); if (!tc_node) return; irdma_ws_cqp_cmd(vsi, tc_node, IRDMA_OP_WS_DELETE_NODE); vsi->unregister_qset(vsi, tc_node); list_del(&tc_node->siblings); irdma_free_node(vsi, tc_node); /* Check if VSI node can be freed */ if (list_empty(&vsi_node->child_list_head)) { irdma_ws_cqp_cmd(vsi, vsi_node, IRDMA_OP_WS_DELETE_NODE); list_del(&vsi_node->siblings); irdma_free_node(vsi, vsi_node); /* Free head node there are no remaining VSI nodes */ if (list_empty(&ws_tree_root->child_list_head)) { irdma_ws_cqp_cmd(vsi, ws_tree_root, IRDMA_OP_WS_DELETE_NODE); irdma_free_node(vsi, ws_tree_root); vsi->dev->ws_tree_root = NULL; } } } /** * irdma_ws_add - Build work scheduler tree, set RDMA qs_handle * @vsi: vsi pointer * @user_pri: user priority */ int irdma_ws_add(struct irdma_sc_vsi *vsi, u8 user_pri) { struct irdma_ws_node *ws_tree_root; struct irdma_ws_node *vsi_node; struct irdma_ws_node *tc_node; u16 traffic_class; int ret = 0; int i; mutex_lock(&vsi->dev->ws_mutex); if (vsi->tc_change_pending) { ret = -EBUSY; goto exit; } if (vsi->qos[user_pri].valid) goto exit; ws_tree_root = vsi->dev->ws_tree_root; if (!ws_tree_root) { ws_tree_root = irdma_alloc_node(vsi, user_pri, WS_NODE_TYPE_PARENT, NULL); if (!ws_tree_root) { ret = -ENOMEM; goto exit; } irdma_debug(vsi->dev, IRDMA_DEBUG_WS, "Creating root node = %d\n", ws_tree_root->index); ret = irdma_ws_cqp_cmd(vsi, ws_tree_root, IRDMA_OP_WS_ADD_NODE); if (ret) { irdma_free_node(vsi, ws_tree_root); goto exit; } vsi->dev->ws_tree_root = ws_tree_root; } /* Find a second tier node that matches the VSI */ vsi_node = ws_find_node(ws_tree_root, vsi->vsi_idx, WS_MATCH_TYPE_VSI); /* If VSI node doesn't exist, add one */ if (!vsi_node) { irdma_debug(vsi->dev, IRDMA_DEBUG_WS, "Node not found matching VSI %d\n", vsi->vsi_idx); vsi_node = irdma_alloc_node(vsi, user_pri, WS_NODE_TYPE_PARENT, ws_tree_root); if (!vsi_node) { ret = -ENOMEM; goto vsi_add_err; } ret = irdma_ws_cqp_cmd(vsi, vsi_node, IRDMA_OP_WS_ADD_NODE); if (ret) { irdma_free_node(vsi, vsi_node); goto vsi_add_err; } list_add(&vsi_node->siblings, &ws_tree_root->child_list_head); } irdma_debug(vsi->dev, IRDMA_DEBUG_WS, "Using node %d which represents VSI %d\n", vsi_node->index, vsi->vsi_idx); traffic_class = vsi->qos[user_pri].traffic_class; tc_node = ws_find_node(vsi_node, traffic_class, WS_MATCH_TYPE_TC); if (!tc_node) { /* Add leaf node */ irdma_debug(vsi->dev, IRDMA_DEBUG_WS, "Node not found matching VSI %d and TC %d\n", vsi->vsi_idx, traffic_class); tc_node = irdma_alloc_node(vsi, user_pri, WS_NODE_TYPE_LEAF, vsi_node); if (!tc_node) { ret = -ENOMEM; goto leaf_add_err; } ret = irdma_ws_cqp_cmd(vsi, tc_node, IRDMA_OP_WS_ADD_NODE); if (ret) { irdma_free_node(vsi, tc_node); goto leaf_add_err; } list_add(&tc_node->siblings, &vsi_node->child_list_head); /* * callback to LAN to update the LAN tree with our node */ ret = vsi->register_qset(vsi, tc_node); if (ret) goto reg_err; tc_node->enable = true; ret = irdma_ws_cqp_cmd(vsi, tc_node, IRDMA_OP_WS_MODIFY_NODE); if (ret) { vsi->unregister_qset(vsi, tc_node); goto reg_err; } } irdma_debug(vsi->dev, IRDMA_DEBUG_WS, "Using node %d which represents VSI %d TC %d\n", tc_node->index, vsi->vsi_idx, traffic_class); /* * Iterate through other UPs and update the QS handle if they have a matching traffic class. */ for (i = 0; i < IRDMA_MAX_USER_PRIORITY; i++) { if (vsi->qos[i].traffic_class == traffic_class) { vsi->qos[i].qs_handle = tc_node->qs_handle; vsi->qos[i].l2_sched_node_id = tc_node->l2_sched_node_id; vsi->qos[i].valid = true; } } goto exit; reg_err: irdma_ws_cqp_cmd(vsi, tc_node, IRDMA_OP_WS_DELETE_NODE); list_del(&tc_node->siblings); irdma_free_node(vsi, tc_node); leaf_add_err: if (list_empty(&vsi_node->child_list_head)) { if (irdma_ws_cqp_cmd(vsi, vsi_node, IRDMA_OP_WS_DELETE_NODE)) goto exit; list_del(&vsi_node->siblings); irdma_free_node(vsi, vsi_node); } vsi_add_err: /* Free head node there are no remaining VSI nodes */ if (list_empty(&ws_tree_root->child_list_head)) { irdma_ws_cqp_cmd(vsi, ws_tree_root, IRDMA_OP_WS_DELETE_NODE); vsi->dev->ws_tree_root = NULL; irdma_free_node(vsi, ws_tree_root); } exit: mutex_unlock(&vsi->dev->ws_mutex); return ret; } /** * irdma_ws_remove - Free WS scheduler node, update WS tree * @vsi: vsi pointer * @user_pri: user priority */ void irdma_ws_remove(struct irdma_sc_vsi *vsi, u8 user_pri) { mutex_lock(&vsi->dev->ws_mutex); if (irdma_ws_in_use(vsi, user_pri)) goto exit; irdma_remove_leaf(vsi, user_pri); exit: mutex_unlock(&vsi->dev->ws_mutex); } /** * irdma_ws_reset - Reset entire WS tree * @vsi: vsi pointer */ void irdma_ws_reset(struct irdma_sc_vsi *vsi) { u8 i; mutex_lock(&vsi->dev->ws_mutex); for (i = 0; i < IRDMA_MAX_USER_PRIORITY; ++i) irdma_remove_leaf(vsi, i); mutex_unlock(&vsi->dev->ws_mutex); }