hc
2024-01-03 2f7c68cb55ecb7331f2381deb497c27155f32faf
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
/* Copyright (c) 2019 Mellanox Technologies */
 
#include <linux/mlx5/driver.h>
#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;
}