162306a36Sopenharmony_ci/* 262306a36Sopenharmony_ci * Copyright (c) 2006, 2007 Cisco Systems, Inc. All rights reserved. 362306a36Sopenharmony_ci * Copyright (c) 2005 Mellanox Technologies. All rights reserved. 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * This software is available to you under a choice of one of two 662306a36Sopenharmony_ci * licenses. You may choose to be licensed under the terms of the GNU 762306a36Sopenharmony_ci * General Public License (GPL) Version 2, available from the file 862306a36Sopenharmony_ci * COPYING in the main directory of this source tree, or the 962306a36Sopenharmony_ci * OpenIB.org BSD license below: 1062306a36Sopenharmony_ci * 1162306a36Sopenharmony_ci * Redistribution and use in source and binary forms, with or 1262306a36Sopenharmony_ci * without modification, are permitted provided that the following 1362306a36Sopenharmony_ci * conditions are met: 1462306a36Sopenharmony_ci * 1562306a36Sopenharmony_ci * - Redistributions of source code must retain the above 1662306a36Sopenharmony_ci * copyright notice, this list of conditions and the following 1762306a36Sopenharmony_ci * disclaimer. 1862306a36Sopenharmony_ci * 1962306a36Sopenharmony_ci * - Redistributions in binary form must reproduce the above 2062306a36Sopenharmony_ci * copyright notice, this list of conditions and the following 2162306a36Sopenharmony_ci * disclaimer in the documentation and/or other materials 2262306a36Sopenharmony_ci * provided with the distribution. 2362306a36Sopenharmony_ci * 2462306a36Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 2562306a36Sopenharmony_ci * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 2662306a36Sopenharmony_ci * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 2762306a36Sopenharmony_ci * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS 2862306a36Sopenharmony_ci * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN 2962306a36Sopenharmony_ci * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN 3062306a36Sopenharmony_ci * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 3162306a36Sopenharmony_ci * SOFTWARE. 3262306a36Sopenharmony_ci */ 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci#include <linux/errno.h> 3562306a36Sopenharmony_ci#include <linux/export.h> 3662306a36Sopenharmony_ci#include <linux/io-mapping.h> 3762306a36Sopenharmony_ci 3862306a36Sopenharmony_ci#include <asm/page.h> 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci#include "mlx4.h" 4162306a36Sopenharmony_ci#include "icm.h" 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_cienum { 4462306a36Sopenharmony_ci MLX4_NUM_RESERVED_UARS = 8 4562306a36Sopenharmony_ci}; 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_ciint mlx4_pd_alloc(struct mlx4_dev *dev, u32 *pdn) 4862306a36Sopenharmony_ci{ 4962306a36Sopenharmony_ci struct mlx4_priv *priv = mlx4_priv(dev); 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_ci *pdn = mlx4_bitmap_alloc(&priv->pd_bitmap); 5262306a36Sopenharmony_ci if (*pdn == -1) 5362306a36Sopenharmony_ci return -ENOMEM; 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_ci return 0; 5662306a36Sopenharmony_ci} 5762306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mlx4_pd_alloc); 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_civoid mlx4_pd_free(struct mlx4_dev *dev, u32 pdn) 6062306a36Sopenharmony_ci{ 6162306a36Sopenharmony_ci mlx4_bitmap_free(&mlx4_priv(dev)->pd_bitmap, pdn, MLX4_USE_RR); 6262306a36Sopenharmony_ci} 6362306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mlx4_pd_free); 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ciint __mlx4_xrcd_alloc(struct mlx4_dev *dev, u32 *xrcdn) 6662306a36Sopenharmony_ci{ 6762306a36Sopenharmony_ci struct mlx4_priv *priv = mlx4_priv(dev); 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci *xrcdn = mlx4_bitmap_alloc(&priv->xrcd_bitmap); 7062306a36Sopenharmony_ci if (*xrcdn == -1) 7162306a36Sopenharmony_ci return -ENOMEM; 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci return 0; 7462306a36Sopenharmony_ci} 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_ciint mlx4_xrcd_alloc(struct mlx4_dev *dev, u32 *xrcdn) 7762306a36Sopenharmony_ci{ 7862306a36Sopenharmony_ci u64 out_param; 7962306a36Sopenharmony_ci int err; 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci if (mlx4_is_mfunc(dev)) { 8262306a36Sopenharmony_ci err = mlx4_cmd_imm(dev, 0, &out_param, 8362306a36Sopenharmony_ci RES_XRCD, RES_OP_RESERVE, 8462306a36Sopenharmony_ci MLX4_CMD_ALLOC_RES, 8562306a36Sopenharmony_ci MLX4_CMD_TIME_CLASS_A, MLX4_CMD_WRAPPED); 8662306a36Sopenharmony_ci if (err) 8762306a36Sopenharmony_ci return err; 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci *xrcdn = get_param_l(&out_param); 9062306a36Sopenharmony_ci return 0; 9162306a36Sopenharmony_ci } 9262306a36Sopenharmony_ci return __mlx4_xrcd_alloc(dev, xrcdn); 9362306a36Sopenharmony_ci} 9462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mlx4_xrcd_alloc); 9562306a36Sopenharmony_ci 9662306a36Sopenharmony_civoid __mlx4_xrcd_free(struct mlx4_dev *dev, u32 xrcdn) 9762306a36Sopenharmony_ci{ 9862306a36Sopenharmony_ci mlx4_bitmap_free(&mlx4_priv(dev)->xrcd_bitmap, xrcdn, MLX4_USE_RR); 9962306a36Sopenharmony_ci} 10062306a36Sopenharmony_ci 10162306a36Sopenharmony_civoid mlx4_xrcd_free(struct mlx4_dev *dev, u32 xrcdn) 10262306a36Sopenharmony_ci{ 10362306a36Sopenharmony_ci u64 in_param = 0; 10462306a36Sopenharmony_ci int err; 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_ci if (mlx4_is_mfunc(dev)) { 10762306a36Sopenharmony_ci set_param_l(&in_param, xrcdn); 10862306a36Sopenharmony_ci err = mlx4_cmd(dev, in_param, RES_XRCD, 10962306a36Sopenharmony_ci RES_OP_RESERVE, MLX4_CMD_FREE_RES, 11062306a36Sopenharmony_ci MLX4_CMD_TIME_CLASS_A, MLX4_CMD_WRAPPED); 11162306a36Sopenharmony_ci if (err) 11262306a36Sopenharmony_ci mlx4_warn(dev, "Failed to release xrcdn %d\n", xrcdn); 11362306a36Sopenharmony_ci } else 11462306a36Sopenharmony_ci __mlx4_xrcd_free(dev, xrcdn); 11562306a36Sopenharmony_ci} 11662306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mlx4_xrcd_free); 11762306a36Sopenharmony_ci 11862306a36Sopenharmony_ciint mlx4_init_pd_table(struct mlx4_dev *dev) 11962306a36Sopenharmony_ci{ 12062306a36Sopenharmony_ci struct mlx4_priv *priv = mlx4_priv(dev); 12162306a36Sopenharmony_ci 12262306a36Sopenharmony_ci return mlx4_bitmap_init(&priv->pd_bitmap, dev->caps.num_pds, 12362306a36Sopenharmony_ci (1 << NOT_MASKED_PD_BITS) - 1, 12462306a36Sopenharmony_ci dev->caps.reserved_pds, 0); 12562306a36Sopenharmony_ci} 12662306a36Sopenharmony_ci 12762306a36Sopenharmony_civoid mlx4_cleanup_pd_table(struct mlx4_dev *dev) 12862306a36Sopenharmony_ci{ 12962306a36Sopenharmony_ci mlx4_bitmap_cleanup(&mlx4_priv(dev)->pd_bitmap); 13062306a36Sopenharmony_ci} 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ciint mlx4_init_xrcd_table(struct mlx4_dev *dev) 13362306a36Sopenharmony_ci{ 13462306a36Sopenharmony_ci struct mlx4_priv *priv = mlx4_priv(dev); 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_ci return mlx4_bitmap_init(&priv->xrcd_bitmap, (1 << 16), 13762306a36Sopenharmony_ci (1 << 16) - 1, dev->caps.reserved_xrcds + 1, 0); 13862306a36Sopenharmony_ci} 13962306a36Sopenharmony_ci 14062306a36Sopenharmony_civoid mlx4_cleanup_xrcd_table(struct mlx4_dev *dev) 14162306a36Sopenharmony_ci{ 14262306a36Sopenharmony_ci mlx4_bitmap_cleanup(&mlx4_priv(dev)->xrcd_bitmap); 14362306a36Sopenharmony_ci} 14462306a36Sopenharmony_ci 14562306a36Sopenharmony_ciint mlx4_uar_alloc(struct mlx4_dev *dev, struct mlx4_uar *uar) 14662306a36Sopenharmony_ci{ 14762306a36Sopenharmony_ci int offset; 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_ci uar->index = mlx4_bitmap_alloc(&mlx4_priv(dev)->uar_table.bitmap); 15062306a36Sopenharmony_ci if (uar->index == -1) 15162306a36Sopenharmony_ci return -ENOMEM; 15262306a36Sopenharmony_ci 15362306a36Sopenharmony_ci if (mlx4_is_slave(dev)) 15462306a36Sopenharmony_ci offset = uar->index % ((int)pci_resource_len(dev->persist->pdev, 15562306a36Sopenharmony_ci 2) / 15662306a36Sopenharmony_ci dev->caps.uar_page_size); 15762306a36Sopenharmony_ci else 15862306a36Sopenharmony_ci offset = uar->index; 15962306a36Sopenharmony_ci uar->pfn = (pci_resource_start(dev->persist->pdev, 2) >> PAGE_SHIFT) 16062306a36Sopenharmony_ci + offset; 16162306a36Sopenharmony_ci uar->map = NULL; 16262306a36Sopenharmony_ci return 0; 16362306a36Sopenharmony_ci} 16462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mlx4_uar_alloc); 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_civoid mlx4_uar_free(struct mlx4_dev *dev, struct mlx4_uar *uar) 16762306a36Sopenharmony_ci{ 16862306a36Sopenharmony_ci mlx4_bitmap_free(&mlx4_priv(dev)->uar_table.bitmap, uar->index, MLX4_USE_RR); 16962306a36Sopenharmony_ci} 17062306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mlx4_uar_free); 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_ciint mlx4_bf_alloc(struct mlx4_dev *dev, struct mlx4_bf *bf, int node) 17362306a36Sopenharmony_ci{ 17462306a36Sopenharmony_ci struct mlx4_priv *priv = mlx4_priv(dev); 17562306a36Sopenharmony_ci struct mlx4_uar *uar; 17662306a36Sopenharmony_ci int err = 0; 17762306a36Sopenharmony_ci int idx; 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci if (!priv->bf_mapping) 18062306a36Sopenharmony_ci return -ENOMEM; 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_ci mutex_lock(&priv->bf_mutex); 18362306a36Sopenharmony_ci if (!list_empty(&priv->bf_list)) 18462306a36Sopenharmony_ci uar = list_entry(priv->bf_list.next, struct mlx4_uar, bf_list); 18562306a36Sopenharmony_ci else { 18662306a36Sopenharmony_ci if (mlx4_bitmap_avail(&priv->uar_table.bitmap) < MLX4_NUM_RESERVED_UARS) { 18762306a36Sopenharmony_ci err = -ENOMEM; 18862306a36Sopenharmony_ci goto out; 18962306a36Sopenharmony_ci } 19062306a36Sopenharmony_ci uar = kmalloc_node(sizeof(*uar), GFP_KERNEL, node); 19162306a36Sopenharmony_ci if (!uar) { 19262306a36Sopenharmony_ci uar = kmalloc(sizeof(*uar), GFP_KERNEL); 19362306a36Sopenharmony_ci if (!uar) { 19462306a36Sopenharmony_ci err = -ENOMEM; 19562306a36Sopenharmony_ci goto out; 19662306a36Sopenharmony_ci } 19762306a36Sopenharmony_ci } 19862306a36Sopenharmony_ci err = mlx4_uar_alloc(dev, uar); 19962306a36Sopenharmony_ci if (err) 20062306a36Sopenharmony_ci goto free_kmalloc; 20162306a36Sopenharmony_ci 20262306a36Sopenharmony_ci uar->map = ioremap(uar->pfn << PAGE_SHIFT, PAGE_SIZE); 20362306a36Sopenharmony_ci if (!uar->map) { 20462306a36Sopenharmony_ci err = -ENOMEM; 20562306a36Sopenharmony_ci goto free_uar; 20662306a36Sopenharmony_ci } 20762306a36Sopenharmony_ci 20862306a36Sopenharmony_ci uar->bf_map = io_mapping_map_wc(priv->bf_mapping, 20962306a36Sopenharmony_ci uar->index << PAGE_SHIFT, 21062306a36Sopenharmony_ci PAGE_SIZE); 21162306a36Sopenharmony_ci if (!uar->bf_map) { 21262306a36Sopenharmony_ci err = -ENOMEM; 21362306a36Sopenharmony_ci goto unamp_uar; 21462306a36Sopenharmony_ci } 21562306a36Sopenharmony_ci uar->free_bf_bmap = 0; 21662306a36Sopenharmony_ci list_add(&uar->bf_list, &priv->bf_list); 21762306a36Sopenharmony_ci } 21862306a36Sopenharmony_ci 21962306a36Sopenharmony_ci idx = ffz(uar->free_bf_bmap); 22062306a36Sopenharmony_ci uar->free_bf_bmap |= 1 << idx; 22162306a36Sopenharmony_ci bf->uar = uar; 22262306a36Sopenharmony_ci bf->offset = 0; 22362306a36Sopenharmony_ci bf->buf_size = dev->caps.bf_reg_size / 2; 22462306a36Sopenharmony_ci bf->reg = uar->bf_map + idx * dev->caps.bf_reg_size; 22562306a36Sopenharmony_ci if (uar->free_bf_bmap == (1 << dev->caps.bf_regs_per_page) - 1) 22662306a36Sopenharmony_ci list_del_init(&uar->bf_list); 22762306a36Sopenharmony_ci 22862306a36Sopenharmony_ci goto out; 22962306a36Sopenharmony_ci 23062306a36Sopenharmony_ciunamp_uar: 23162306a36Sopenharmony_ci bf->uar = NULL; 23262306a36Sopenharmony_ci iounmap(uar->map); 23362306a36Sopenharmony_ci 23462306a36Sopenharmony_cifree_uar: 23562306a36Sopenharmony_ci mlx4_uar_free(dev, uar); 23662306a36Sopenharmony_ci 23762306a36Sopenharmony_cifree_kmalloc: 23862306a36Sopenharmony_ci kfree(uar); 23962306a36Sopenharmony_ci 24062306a36Sopenharmony_ciout: 24162306a36Sopenharmony_ci mutex_unlock(&priv->bf_mutex); 24262306a36Sopenharmony_ci return err; 24362306a36Sopenharmony_ci} 24462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mlx4_bf_alloc); 24562306a36Sopenharmony_ci 24662306a36Sopenharmony_civoid mlx4_bf_free(struct mlx4_dev *dev, struct mlx4_bf *bf) 24762306a36Sopenharmony_ci{ 24862306a36Sopenharmony_ci struct mlx4_priv *priv = mlx4_priv(dev); 24962306a36Sopenharmony_ci int idx; 25062306a36Sopenharmony_ci 25162306a36Sopenharmony_ci if (!bf->uar || !bf->uar->bf_map) 25262306a36Sopenharmony_ci return; 25362306a36Sopenharmony_ci 25462306a36Sopenharmony_ci mutex_lock(&priv->bf_mutex); 25562306a36Sopenharmony_ci idx = (bf->reg - bf->uar->bf_map) / dev->caps.bf_reg_size; 25662306a36Sopenharmony_ci bf->uar->free_bf_bmap &= ~(1 << idx); 25762306a36Sopenharmony_ci if (!bf->uar->free_bf_bmap) { 25862306a36Sopenharmony_ci if (!list_empty(&bf->uar->bf_list)) 25962306a36Sopenharmony_ci list_del(&bf->uar->bf_list); 26062306a36Sopenharmony_ci 26162306a36Sopenharmony_ci io_mapping_unmap(bf->uar->bf_map); 26262306a36Sopenharmony_ci iounmap(bf->uar->map); 26362306a36Sopenharmony_ci mlx4_uar_free(dev, bf->uar); 26462306a36Sopenharmony_ci kfree(bf->uar); 26562306a36Sopenharmony_ci } else if (list_empty(&bf->uar->bf_list)) 26662306a36Sopenharmony_ci list_add(&bf->uar->bf_list, &priv->bf_list); 26762306a36Sopenharmony_ci 26862306a36Sopenharmony_ci mutex_unlock(&priv->bf_mutex); 26962306a36Sopenharmony_ci} 27062306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mlx4_bf_free); 27162306a36Sopenharmony_ci 27262306a36Sopenharmony_ciint mlx4_init_uar_table(struct mlx4_dev *dev) 27362306a36Sopenharmony_ci{ 27462306a36Sopenharmony_ci int num_reserved_uar = mlx4_get_num_reserved_uar(dev); 27562306a36Sopenharmony_ci 27662306a36Sopenharmony_ci mlx4_dbg(dev, "uar_page_shift = %d", dev->uar_page_shift); 27762306a36Sopenharmony_ci mlx4_dbg(dev, "Effective reserved_uars=%d", dev->caps.reserved_uars); 27862306a36Sopenharmony_ci 27962306a36Sopenharmony_ci if (dev->caps.num_uars <= num_reserved_uar) { 28062306a36Sopenharmony_ci mlx4_err( 28162306a36Sopenharmony_ci dev, "Only %d UAR pages (need more than %d)\n", 28262306a36Sopenharmony_ci dev->caps.num_uars, num_reserved_uar); 28362306a36Sopenharmony_ci mlx4_err(dev, "Increase firmware log2_uar_bar_megabytes?\n"); 28462306a36Sopenharmony_ci return -ENODEV; 28562306a36Sopenharmony_ci } 28662306a36Sopenharmony_ci 28762306a36Sopenharmony_ci return mlx4_bitmap_init(&mlx4_priv(dev)->uar_table.bitmap, 28862306a36Sopenharmony_ci dev->caps.num_uars, dev->caps.num_uars - 1, 28962306a36Sopenharmony_ci dev->caps.reserved_uars, 0); 29062306a36Sopenharmony_ci} 29162306a36Sopenharmony_ci 29262306a36Sopenharmony_civoid mlx4_cleanup_uar_table(struct mlx4_dev *dev) 29362306a36Sopenharmony_ci{ 29462306a36Sopenharmony_ci mlx4_bitmap_cleanup(&mlx4_priv(dev)->uar_table.bitmap); 29562306a36Sopenharmony_ci} 296