Author: hselasky
Date: Thu Mar  8 11:59:47 2018
New Revision: 330651
URL: https://svnweb.freebsd.org/changeset/base/330651

Log:
  Add vendor specific capability interface support in mlx5core.
  
  Add the ability to access the vendor specific space gateway in order
  to support reading and writing data into the different configuration
  domains.
  
  Submitted by: Matthew Finlay <m...@mellanox.com>
  MFC after:    1 week
  Sponsored by: Mellanox Technologies

Added:
  head/sys/dev/mlx5/mlx5_core/mlx5_vsc.c   (contents, props changed)
Modified:
  head/sys/conf/files
  head/sys/dev/mlx5/device.h
  head/sys/dev/mlx5/driver.h
  head/sys/dev/mlx5/mlx5_core/mlx5_main.c
  head/sys/dev/mlx5/mlx5_ifc.h
  head/sys/modules/mlx5/Makefile

Modified: head/sys/conf/files
==============================================================================
--- head/sys/conf/files Thu Mar  8 11:58:27 2018        (r330650)
+++ head/sys/conf/files Thu Mar  8 11:59:47 2018        (r330651)
@@ -4757,6 +4757,8 @@ dev/mlx5/mlx5_core/mlx5_uar.c                     
optional mlx5 pci       \
        compile-with "${OFED_C}"
 dev/mlx5/mlx5_core/mlx5_vport.c                        optional mlx5 pci       
\
        compile-with "${OFED_C}"
+dev/mlx5/mlx5_core/mlx5_vsc.c                  optional mlx5 pci       \
+       compile-with "${OFED_C}"
 dev/mlx5/mlx5_core/mlx5_wq.c                   optional mlx5 pci       \
        compile-with "${OFED_C}"
 

Modified: head/sys/dev/mlx5/device.h
==============================================================================
--- head/sys/dev/mlx5/device.h  Thu Mar  8 11:58:27 2018        (r330650)
+++ head/sys/dev/mlx5/device.h  Thu Mar  8 11:59:47 2018        (r330651)
@@ -138,6 +138,26 @@ __mlx5_mask(typ, fld))
                                     MLX5_BY_PASS_NUM_DONT_TRAP_PRIOS +\
                                     MLX5_BY_PASS_NUM_MULTICAST_PRIOS)
 
+/* insert a value to a struct */
+#define MLX5_VSC_SET(typ, p, fld, v) do { \
+       BUILD_BUG_ON(__mlx5_st_sz_bits(typ) % 32);             \
+       BUILD_BUG_ON(__mlx5_bit_sz(typ, fld) > 32); \
+       *((__le32 *)(p) + __mlx5_dw_off(typ, fld)) = \
+       cpu_to_le32((le32_to_cpu(*((__le32 *)(p) + __mlx5_dw_off(typ, fld))) & \
+                    (~__mlx5_dw_mask(typ, fld))) | (((v) & __mlx5_mask(typ, 
fld)) \
+                    << __mlx5_dw_bit_off(typ, fld))); \
+} while (0)
+
+#define MLX5_VSC_GET(typ, p, fld) ((le32_to_cpu(*((__le32 *)(p) +\
+__mlx5_dw_off(typ, fld))) >> __mlx5_dw_bit_off(typ, fld)) & \
+__mlx5_mask(typ, fld))
+
+#define MLX5_VSC_GET_PR(typ, p, fld) ({ \
+       u32 ___t = MLX5_VSC_GET(typ, p, fld); \
+       pr_debug(#fld " = 0x%x\n", ___t); \
+       ___t; \
+})
+
 enum {
        MLX5_MAX_COMMANDS               = 32,
        MLX5_CMD_DATA_BLOCK_SIZE        = 512,

Modified: head/sys/dev/mlx5/driver.h
==============================================================================
--- head/sys/dev/mlx5/driver.h  Thu Mar  8 11:58:27 2018        (r330650)
+++ head/sys/dev/mlx5/driver.h  Thu Mar  8 11:59:47 2018        (r330651)
@@ -221,6 +221,16 @@ enum mlx5_link_mode {
        MLX5_LINK_MODES_NUMBER,
 };
 
+enum {
+       MLX5_VSC_SPACE_SUPPORTED = 0x1,
+       MLX5_VSC_SPACE_OFFSET    = 0x4,
+       MLX5_VSC_COUNTER_OFFSET  = 0x8,
+       MLX5_VSC_SEMA_OFFSET     = 0xC,
+       MLX5_VSC_ADDR_OFFSET     = 0x10,
+       MLX5_VSC_DATA_OFFSET     = 0x14,
+       MLX5_VSC_MAX_RETRIES     = 0x1000,
+};
+
 #define MLX5_PROT_MASK(link_mode) (1 << link_mode)
 
 struct mlx5_uuar_info {
@@ -627,6 +637,7 @@ struct mlx5_core_dev {
        struct mlx5_priv        priv;
        struct mlx5_profile     *profile;
        atomic_t                num_qps;
+       u32                     vsc_addr;
        u32                     issi;
        struct mlx5_special_contexts special_contexts;
        unsigned int module_status[MLX5_MAX_PORTS];
@@ -1001,6 +1012,12 @@ int mlx5_set_diagnostic_params(struct mlx5_core_dev *m
 int mlx5_query_diagnostic_counters(struct mlx5_core_dev *mdev,
                                   u8 num_of_samples, u16 sample_index,
                                   void *out, int out_size);
+int mlx5_vsc_find_cap(struct mlx5_core_dev *mdev);
+int mlx5_vsc_lock(struct mlx5_core_dev *mdev);
+void mlx5_vsc_unlock(struct mlx5_core_dev *mdev);
+int mlx5_vsc_set_space(struct mlx5_core_dev *mdev, u16 space);
+int mlx5_vsc_write(struct mlx5_core_dev *mdev, u32 addr, u32 *data);
+int mlx5_vsc_read(struct mlx5_core_dev *mdev, u32 addr, u32 *data);
 static inline u32 mlx5_mkey_to_idx(u32 mkey)
 {
        return mkey >> 8;

Modified: head/sys/dev/mlx5/mlx5_core/mlx5_main.c
==============================================================================
--- head/sys/dev/mlx5/mlx5_core/mlx5_main.c     Thu Mar  8 11:58:27 2018        
(r330650)
+++ head/sys/dev/mlx5/mlx5_core/mlx5_main.c     Thu Mar  8 11:59:47 2018        
(r330651)
@@ -846,6 +846,9 @@ static int mlx5_pci_init(struct mlx5_core_dev *dev, st
                goto err_clr_master;
        }
 
+       if (mlx5_vsc_find_cap(dev))
+               dev_err(&pdev->dev, "Unable to find vendor specific 
capabilities\n");
+
         return 0;
 
 err_clr_master:
@@ -1183,7 +1186,6 @@ static int init_one(struct pci_dev *pdev,
                device_printf((&pdev->dev)->bsddev, "ERR: 
""mlx5_register_device failed %d\n", err);
                goto clean_health;
        }
-
 
        return 0;
 

Added: head/sys/dev/mlx5/mlx5_core/mlx5_vsc.c
==============================================================================
--- /dev/null   00:00:00 1970   (empty, because file is newly added)
+++ head/sys/dev/mlx5/mlx5_core/mlx5_vsc.c      Thu Mar  8 11:59:47 2018        
(r330651)
@@ -0,0 +1,204 @@
+/*-
+ * Copyright (c) 2013-2017, Mellanox Technologies, Ltd.         All rights 
reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. 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.
+ *
+ * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS `AS IS' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ * $FreeBSD$
+ */
+
+#include <dev/mlx5/driver.h>
+#include <dev/mlx5/device.h>
+#include <dev/mlx5/mlx5_core/mlx5_core.h>
+
+struct mlx5_ifc_vsc_space_bits {
+       u8 status[0x3];
+       u8 reserved0[0xd];
+       u8 space[0x10];
+};
+
+struct mlx5_ifc_vsc_addr_bits {
+       u8 flag[0x1];
+       u8 reserved0[0x1];
+       u8 address[0x1e];
+};
+
+int mlx5_vsc_lock(struct mlx5_core_dev *mdev)
+{
+       device_t dev = mdev->pdev->dev.bsddev;
+       int vsc_addr = mdev->vsc_addr;
+       int retries = 0;
+       u32 lock_val;
+       u32 counter;
+
+       if (!vsc_addr) {
+               mlx5_core_warn(mdev, "Unable to acquire vsc lock, vsc_addr not 
initialized\n");
+               return EINVAL;
+       }
+
+       while (true) {
+               if (retries > MLX5_VSC_MAX_RETRIES)
+                       return EBUSY;
+
+               if (pci_read_config(dev, vsc_addr + MLX5_VSC_SEMA_OFFSET, 4)) {
+                       retries++;
+                       /*
+                        * The PRM suggests random 0 - 10ms to prevent multiple
+                        * waiters on the same interval in order to avoid 
starvation
+                        */
+                       DELAY((random() % 11) * 1000);
+                       continue;
+               }
+
+               counter = pci_read_config(dev, vsc_addr + 
MLX5_VSC_COUNTER_OFFSET, 4);
+               pci_write_config(dev, vsc_addr + MLX5_VSC_SEMA_OFFSET, counter, 
4);
+               lock_val = pci_read_config(dev, vsc_addr + 
MLX5_VSC_SEMA_OFFSET, 4);
+
+               if (lock_val == counter)
+                       break;
+
+               retries++;
+       }
+
+       return 0;
+}
+
+void mlx5_vsc_unlock(struct mlx5_core_dev *mdev)
+{
+       device_t dev = mdev->pdev->dev.bsddev;
+       int vsc_addr = mdev->vsc_addr;
+
+       if (!vsc_addr) {
+               mlx5_core_warn(mdev, "Unable to release vsc lock, vsc_addr not 
initialized\n");
+               return;
+       }
+
+       pci_write_config(dev, vsc_addr + MLX5_VSC_SEMA_OFFSET, 0, 4);
+}
+
+static int mlx5_vsc_wait_on_flag(struct mlx5_core_dev *mdev, u32 expected)
+{
+       device_t dev = mdev->pdev->dev.bsddev;
+       int vsc_addr = mdev->vsc_addr;
+       int retries = 0;
+       u32 flag;
+
+       while (true) {
+               if (retries > MLX5_VSC_MAX_RETRIES)
+                       return EBUSY;
+
+               flag = pci_read_config(dev, vsc_addr + MLX5_VSC_ADDR_OFFSET, 4);
+               if (expected == MLX5_VSC_GET(vsc_addr, &flag, flag))
+                       break;
+
+               retries++;
+               DELAY(10);
+       }
+
+       return 0;
+}
+
+int mlx5_vsc_set_space(struct mlx5_core_dev *mdev, u16 space)
+{
+       device_t dev = mdev->pdev->dev.bsddev;
+       int vsc_addr = mdev->vsc_addr;
+       u32 vsc_space = 0;
+
+       if (!vsc_addr) {
+               mlx5_core_warn(mdev, "Unable to set vsc space, vsc_addr not 
initialized\n");
+               return EINVAL;
+       }
+
+       MLX5_VSC_SET(vsc_space, &vsc_space, space, space);
+       pci_write_config(dev, vsc_addr + MLX5_VSC_SPACE_OFFSET, vsc_space, 4);
+       vsc_space = pci_read_config(dev, vsc_addr + MLX5_VSC_SPACE_OFFSET, 4);
+
+       if (MLX5_VSC_GET(vsc_space, &vsc_space, status) != 
MLX5_VSC_SPACE_SUPPORTED) {
+               mlx5_core_warn(mdev, "Space 0x%x is not supported.\n", space);
+               return ENOTSUP;
+       }
+
+       return 0;
+}
+
+int mlx5_vsc_write(struct mlx5_core_dev *mdev, u32 addr, u32 *data)
+{
+       device_t dev = mdev->pdev->dev.bsddev;
+       int vsc_addr = mdev->vsc_addr;
+       u32 in = 0;
+       int err;
+
+       if (!vsc_addr) {
+               mlx5_core_warn(mdev, "Unable to call vsc write, vsc_addr not 
initialized\n");
+               return EINVAL;
+       }
+
+       MLX5_VSC_SET(vsc_addr, &in, address, addr);
+       MLX5_VSC_SET(vsc_addr, &in, flag, 1);
+       pci_write_config(dev, vsc_addr + MLX5_VSC_DATA_OFFSET, *data, 4);
+       pci_write_config(dev, vsc_addr + MLX5_VSC_ADDR_OFFSET, in, 4);
+
+       err = mlx5_vsc_wait_on_flag(mdev, 0);
+       if (err)
+               mlx5_core_warn(mdev, "Failed waiting for write flag!\n");
+
+       return err;
+}
+
+int mlx5_vsc_read(struct mlx5_core_dev *mdev, u32 addr, u32 *data)
+{
+       device_t dev = mdev->pdev->dev.bsddev;
+       int vsc_addr = mdev->vsc_addr;
+       int err;
+       u32 in;
+
+       if (!vsc_addr) {
+               mlx5_core_warn(mdev, "Unable to call vsc read, vsc_addr not 
initialized\n");
+               return EINVAL;
+       }
+
+       MLX5_VSC_SET(vsc_addr, &in, address, addr);
+       pci_write_config(dev, vsc_addr + MLX5_VSC_ADDR_OFFSET, in, 4);
+
+       err = mlx5_vsc_wait_on_flag(mdev, 1);
+       if (err) {
+               mlx5_core_warn(mdev, "Failed waiting for read complete 
flag!\n");
+               return err;
+       }
+
+       *data = pci_read_config(dev, vsc_addr + MLX5_VSC_DATA_OFFSET, 4);
+
+       return 0;
+}
+
+int mlx5_vsc_find_cap(struct mlx5_core_dev *mdev)
+{
+       int *capreg = &mdev->vsc_addr;
+       int err;
+
+       err = pci_find_cap(mdev->pdev->dev.bsddev, PCIY_VENDOR, capreg);
+
+       if (err)
+               *capreg = 0;
+
+       return err;
+}
+

Modified: head/sys/dev/mlx5/mlx5_ifc.h
==============================================================================
--- head/sys/dev/mlx5/mlx5_ifc.h        Thu Mar  8 11:58:27 2018        
(r330650)
+++ head/sys/dev/mlx5/mlx5_ifc.h        Thu Mar  8 11:59:47 2018        
(r330651)
@@ -9328,6 +9328,13 @@ struct mlx5_ifc_mtt_bits {
        u8         rd_en[0x1];
 };
 
+/* Vendor Specific Capabilities, VSC */
+enum {
+       MLX5_VSC_DOMAIN_ICMD                    = 0x1,
+       MLX5_VSC_DOMAIN_PROTECTED_CRSPACE       = 0x6,
+       MLX5_VSC_DOMAIN_SEMAPHORES              = 0xA,
+};
+
 struct mlx5_ifc_vendor_specific_cap_bits {
        u8         type[0x8];
        u8         length[0x8];

Modified: head/sys/modules/mlx5/Makefile
==============================================================================
--- head/sys/modules/mlx5/Makefile      Thu Mar  8 11:58:27 2018        
(r330650)
+++ head/sys/modules/mlx5/Makefile      Thu Mar  8 11:59:47 2018        
(r330651)
@@ -24,6 +24,7 @@ mlx5_srq.c \
 mlx5_transobj.c \
 mlx5_uar.c \
 mlx5_vport.c \
+mlx5_vsc.c \
 mlx5_wq.c \
 device_if.h bus_if.h vnode_if.h pci_if.h \
         opt_inet.h opt_inet6.h opt_rss.h
_______________________________________________
svn-src-head@freebsd.org mailing list
https://lists.freebsd.org/mailman/listinfo/svn-src-head
To unsubscribe, send any mail to "svn-src-head-unsubscr...@freebsd.org"

Reply via email to