// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB /* Copyright (c) 2019 Mellanox Technologies */ #include #include "mlx5_core.h" #include "lib/pci_vsc.h" #include "lib/mlx5.h" #define BAD_ACCESS 0xBADACCE5 #define MLX5_PROTECTED_CR_SCAN_CRSPACE 0x7 static bool mlx5_crdump_enabled(struct mlx5_core_dev *dev) { return !!dev->priv.health.crdump_size; } static int mlx5_crdump_fill(struct mlx5_core_dev *dev, u32 *cr_data) { u32 crdump_size = dev->priv.health.crdump_size; int i, ret; for (i = 0; i < (crdump_size / 4); i++) cr_data[i] = BAD_ACCESS; ret = mlx5_vsc_gw_read_block_fast(dev, cr_data, crdump_size); if (ret <= 0) { if (ret == 0) return -EIO; return ret; } if (crdump_size != ret) { mlx5_core_warn(dev, "failed to read full dump, read %d out of %u\n", ret, crdump_size); return -EINVAL; } return 0; } int mlx5_crdump_collect(struct mlx5_core_dev *dev, u32 *cr_data) { int ret; if (!mlx5_crdump_enabled(dev)) return -ENODEV; ret = mlx5_vsc_gw_lock(dev); if (ret) { mlx5_core_warn(dev, "crdump: failed to lock vsc gw err %d\n", ret); return ret; } /* Verify no other PF is running cr-dump or sw reset */ ret = mlx5_vsc_sem_set_space(dev, MLX5_SEMAPHORE_SW_RESET, MLX5_VSC_LOCK); if (ret) { mlx5_core_warn(dev, "Failed to lock SW reset semaphore\n"); goto unlock_gw; } ret = mlx5_vsc_gw_set_space(dev, MLX5_VSC_SPACE_SCAN_CRSPACE, NULL); if (ret) goto unlock_sem; ret = mlx5_crdump_fill(dev, cr_data); unlock_sem: mlx5_vsc_sem_set_space(dev, MLX5_SEMAPHORE_SW_RESET, MLX5_VSC_UNLOCK); unlock_gw: mlx5_vsc_gw_unlock(dev); return ret; } int mlx5_crdump_enable(struct mlx5_core_dev *dev) { struct mlx5_priv *priv = &dev->priv; u32 space_size; int ret; if (!mlx5_core_is_pf(dev) || !mlx5_vsc_accessible(dev) || mlx5_crdump_enabled(dev)) return 0; ret = mlx5_vsc_gw_lock(dev); if (ret) return ret; /* Check if space is supported and get space size */ ret = mlx5_vsc_gw_set_space(dev, MLX5_VSC_SPACE_SCAN_CRSPACE, &space_size); if (ret) { /* Unlock and mask error since space is not supported */ mlx5_vsc_gw_unlock(dev); return 0; } if (!space_size) { mlx5_core_warn(dev, "Invalid Crspace size, zero\n"); mlx5_vsc_gw_unlock(dev); return -EINVAL; } ret = mlx5_vsc_gw_unlock(dev); if (ret) return ret; priv->health.crdump_size = space_size; return 0; } void mlx5_crdump_disable(struct mlx5_core_dev *dev) { dev->priv.health.crdump_size = 0; }