diff mbox series

[v3,net-next,02/11] qed/qede: make devlink survive recovery

Message ID 20200729113846.1551-3-irusskikh@marvell.com
State Changes Requested
Delegated to: David Miller
Headers show
Series qed: introduce devlink health support | expand

Commit Message

Igor Russkikh July 29, 2020, 11:38 a.m. UTC
Before that, devlink instance lifecycle was linked to qed_dev object,
that causes devlink to be recreated on each recovery.

Changing it by making higher level driver (qede) responsible for its
life. This way devlink will survive recoveries.

qede will store devlink structure pointer as a part of its device
object, devlink private data contains a linkage structure, it'll
contain extra devlink related content in following patches.

The same lifecycle should be applied to storage drivers (qedf/qedi) later.

Signed-off-by: Igor Russkikh <irusskikh@marvell.com>
Signed-off-by: Alexander Lobakin <alobakin@marvell.com>
Signed-off-by: Michal Kalderon <michal.kalderon@marvell.com>
---
 drivers/net/ethernet/qlogic/qed/qed.h         |  1 -
 drivers/net/ethernet/qlogic/qed/qed_devlink.c | 40 ++++++++-----------
 drivers/net/ethernet/qlogic/qed/qed_devlink.h |  4 +-
 drivers/net/ethernet/qlogic/qed/qed_main.c    | 10 +----
 drivers/net/ethernet/qlogic/qede/qede.h       |  1 +
 drivers/net/ethernet/qlogic/qede/qede_main.c  | 18 +++++++++
 include/linux/qed/qed_if.h                    |  9 +++++
 7 files changed, 48 insertions(+), 35 deletions(-)

Comments

Jiri Pirko July 29, 2020, 12:49 p.m. UTC | #1
Wed, Jul 29, 2020 at 01:38:37PM CEST, irusskikh@marvell.com wrote:
>Before that, devlink instance lifecycle was linked to qed_dev object,

Before what?


>that causes devlink to be recreated on each recovery.
>
>Changing it by making higher level driver (qede) responsible for its
>life. This way devlink will survive recoveries.
>
>qede will store devlink structure pointer as a part of its device
>object, devlink private data contains a linkage structure, it'll
>contain extra devlink related content in following patches.
>
>The same lifecycle should be applied to storage drivers (qedf/qedi) later.

From this, one can't really tell if you are talking about existing state
or about the matter of this patch. In the patch description, you should
be imperative to the codebase telling it what to do. Could you please do
that?


>
>Signed-off-by: Igor Russkikh <irusskikh@marvell.com>
>Signed-off-by: Alexander Lobakin <alobakin@marvell.com>
>Signed-off-by: Michal Kalderon <michal.kalderon@marvell.com>
>---
> drivers/net/ethernet/qlogic/qed/qed.h         |  1 -
> drivers/net/ethernet/qlogic/qed/qed_devlink.c | 40 ++++++++-----------
> drivers/net/ethernet/qlogic/qed/qed_devlink.h |  4 +-
> drivers/net/ethernet/qlogic/qed/qed_main.c    | 10 +----
> drivers/net/ethernet/qlogic/qede/qede.h       |  1 +
> drivers/net/ethernet/qlogic/qede/qede_main.c  | 18 +++++++++
> include/linux/qed/qed_if.h                    |  9 +++++
> 7 files changed, 48 insertions(+), 35 deletions(-)
>
>diff --git a/drivers/net/ethernet/qlogic/qed/qed.h b/drivers/net/ethernet/qlogic/qed/qed.h
>index b2a7b53ee760..b6ce1488abcc 100644
>--- a/drivers/net/ethernet/qlogic/qed/qed.h
>+++ b/drivers/net/ethernet/qlogic/qed/qed.h
>@@ -849,7 +849,6 @@ struct qed_dev {
> 	u32 rdma_max_srq_sge;
> 	u16 tunn_feature_mask;
> 
>-	struct devlink			*dl;
> 	bool				iwarp_cmt;
> };
> 
>diff --git a/drivers/net/ethernet/qlogic/qed/qed_devlink.c b/drivers/net/ethernet/qlogic/qed/qed_devlink.c
>index eb693787c99e..a62c47c61edf 100644
>--- a/drivers/net/ethernet/qlogic/qed/qed_devlink.c
>+++ b/drivers/net/ethernet/qlogic/qed/qed_devlink.c
>@@ -5,6 +5,7 @@
>  */
> 
> #include <linux/kernel.h>
>+#include <linux/qed/qed_if.h>
> #include "qed.h"
> #include "qed_devlink.h"
> 
>@@ -13,17 +14,12 @@ enum qed_devlink_param_id {
> 	QED_DEVLINK_PARAM_ID_IWARP_CMT,
> };
> 
>-struct qed_devlink {
>-	struct qed_dev *cdev;
>-};
>-
> static int qed_dl_param_get(struct devlink *dl, u32 id,
> 			    struct devlink_param_gset_ctx *ctx)
> {
>-	struct qed_devlink *qed_dl;
>+	struct qed_devlink *qed_dl = devlink_priv(dl);
> 	struct qed_dev *cdev;
> 
>-	qed_dl = devlink_priv(dl);
> 	cdev = qed_dl->cdev;
> 	ctx->val.vbool = cdev->iwarp_cmt;
> 
>@@ -33,10 +29,9 @@ static int qed_dl_param_get(struct devlink *dl, u32 id,
> static int qed_dl_param_set(struct devlink *dl, u32 id,
> 			    struct devlink_param_gset_ctx *ctx)
> {
>-	struct qed_devlink *qed_dl;
>+	struct qed_devlink *qed_dl = devlink_priv(dl);
> 	struct qed_dev *cdev;
> 
>-	qed_dl = devlink_priv(dl);
> 	cdev = qed_dl->cdev;
> 	cdev->iwarp_cmt = ctx->val.vbool;
> 
>@@ -52,21 +47,19 @@ static const struct devlink_param qed_devlink_params[] = {
> 
> static const struct devlink_ops qed_dl_ops;
> 
>-int qed_devlink_register(struct qed_dev *cdev)
>+struct devlink *qed_devlink_register(struct qed_dev *cdev)
> {
> 	union devlink_param_value value;
>-	struct qed_devlink *qed_dl;
>+	struct qed_devlink *qdevlink;
> 	struct devlink *dl;
> 	int rc;
> 
>-	dl = devlink_alloc(&qed_dl_ops, sizeof(*qed_dl));
>+	dl = devlink_alloc(&qed_dl_ops, sizeof(struct qed_devlink));

Do "sizeof(*qdevlink)" instead.


> 	if (!dl)
>-		return -ENOMEM;
>+		return ERR_PTR(-ENOMEM);
> 
>-	qed_dl = devlink_priv(dl);
>-
>-	cdev->dl = dl;
>-	qed_dl->cdev = cdev;
>+	qdevlink = devlink_priv(dl);
>+	qdevlink->cdev = cdev;
> 
> 	rc = devlink_register(dl, &cdev->pdev->dev);
> 	if (rc)
>@@ -85,26 +78,25 @@ int qed_devlink_register(struct qed_dev *cdev)
> 	devlink_params_publish(dl);
> 	cdev->iwarp_cmt = false;
> 
>-	return 0;
>+	return dl;
> 
> err_unregister:
> 	devlink_unregister(dl);
> 
> err_free:
>-	cdev->dl = NULL;
> 	devlink_free(dl);
> 
>-	return rc;
>+	return ERR_PTR(rc);
> }
> 
>-void qed_devlink_unregister(struct qed_dev *cdev)
>+void qed_devlink_unregister(struct devlink *devlink)
> {
>-	if (!cdev->dl)
>+	if (!devlink)
> 		return;
> 
>-	devlink_params_unregister(cdev->dl, qed_devlink_params,
>+	devlink_params_unregister(devlink, qed_devlink_params,
> 				  ARRAY_SIZE(qed_devlink_params));
> 
>-	devlink_unregister(cdev->dl);
>-	devlink_free(cdev->dl);
>+	devlink_unregister(devlink);
>+	devlink_free(devlink);
> }
>diff --git a/drivers/net/ethernet/qlogic/qed/qed_devlink.h b/drivers/net/ethernet/qlogic/qed/qed_devlink.h
>index b94c40e9b7c1..c79dc6bfa194 100644
>--- a/drivers/net/ethernet/qlogic/qed/qed_devlink.h
>+++ b/drivers/net/ethernet/qlogic/qed/qed_devlink.h
>@@ -9,7 +9,7 @@
> #include <linux/qed/qed_if.h>
> #include <net/devlink.h>
> 
>-int qed_devlink_register(struct qed_dev *cdev);
>-void qed_devlink_unregister(struct qed_dev *cdev);
>+struct devlink *qed_devlink_register(struct qed_dev *cdev);
>+void qed_devlink_unregister(struct devlink *devlink);
> 
> #endif
>diff --git a/drivers/net/ethernet/qlogic/qed/qed_main.c b/drivers/net/ethernet/qlogic/qed/qed_main.c
>index 8751355d9ef7..d6f76421379b 100644
>--- a/drivers/net/ethernet/qlogic/qed/qed_main.c
>+++ b/drivers/net/ethernet/qlogic/qed/qed_main.c
>@@ -539,12 +539,6 @@ static struct qed_dev *qed_probe(struct pci_dev *pdev,
> 	}
> 	DP_INFO(cdev, "PCI init completed successfully\n");
> 
>-	rc = qed_devlink_register(cdev);
>-	if (rc) {
>-		DP_INFO(cdev, "Failed to register devlink.\n");
>-		goto err2;
>-	}
>-
> 	rc = qed_hw_prepare(cdev, QED_PCI_DEFAULT);
> 	if (rc) {
> 		DP_ERR(cdev, "hw prepare failed\n");
>@@ -574,8 +568,6 @@ static void qed_remove(struct qed_dev *cdev)
> 
> 	qed_set_power_state(cdev, PCI_D3hot);
> 
>-	qed_devlink_unregister(cdev);
>-
> 	qed_free_cdev(cdev);
> }
> 
>@@ -3012,6 +3004,8 @@ const struct qed_common_ops qed_common_ops_pass = {
> 	.get_link = &qed_get_current_link,
> 	.drain = &qed_drain,
> 	.update_msglvl = &qed_init_dp,
>+	.devlink_register = qed_devlink_register,
>+	.devlink_unregister = qed_devlink_unregister,
> 	.dbg_all_data = &qed_dbg_all_data,
> 	.dbg_all_data_size = &qed_dbg_all_data_size,
> 	.chain_alloc = &qed_chain_alloc,
>diff --git a/drivers/net/ethernet/qlogic/qede/qede.h b/drivers/net/ethernet/qlogic/qede/qede.h
>index 803c1fcca8ad..1f0e7505a973 100644
>--- a/drivers/net/ethernet/qlogic/qede/qede.h
>+++ b/drivers/net/ethernet/qlogic/qede/qede.h
>@@ -172,6 +172,7 @@ struct qede_dev {
> 	struct qed_dev			*cdev;
> 	struct net_device		*ndev;
> 	struct pci_dev			*pdev;
>+	struct devlink			*devlink;
> 
> 	u32				dp_module;
> 	u8				dp_level;
>diff --git a/drivers/net/ethernet/qlogic/qede/qede_main.c b/drivers/net/ethernet/qlogic/qede/qede_main.c
>index 1aaae3203f5a..7c2d948b2035 100644
>--- a/drivers/net/ethernet/qlogic/qede/qede_main.c
>+++ b/drivers/net/ethernet/qlogic/qede/qede_main.c
>@@ -1172,10 +1172,23 @@ static int __qede_probe(struct pci_dev *pdev, u32 dp_module, u8 dp_level,
> 			rc = -ENOMEM;
> 			goto err2;
> 		}
>+
>+		edev->devlink = qed_ops->common->devlink_register(cdev);
>+		if (IS_ERR(edev->devlink)) {
>+			DP_NOTICE(edev, "Cannot register devlink\n");
>+			edev->devlink = NULL;
>+			/* Go on, we can live without devlink */

Interesting. Why? This can happen only very pressing memory
circumstances. In that case, you should just fail to init. 


>+		}
> 	} else {
> 		struct net_device *ndev = pci_get_drvdata(pdev);
> 
> 		edev = netdev_priv(ndev);
>+
>+		if (edev && edev->devlink) {
>+			struct qed_devlink *qdl = devlink_priv(edev->devlink);
>+
>+			qdl->cdev = cdev;
>+		}
> 		edev->cdev = cdev;
> 		memset(&edev->stats, 0, sizeof(edev->stats));
> 		memcpy(&edev->dev_info, &dev_info, sizeof(dev_info));
>@@ -1298,6 +1311,11 @@ static void __qede_remove(struct pci_dev *pdev, enum qede_remove_mode mode)
> 	qed_ops->common->slowpath_stop(cdev);
> 	if (system_state == SYSTEM_POWER_OFF)
> 		return;
>+
>+	if (mode != QEDE_REMOVE_RECOVERY && edev->devlink) {
>+		qed_ops->common->devlink_unregister(edev->devlink);
>+		edev->devlink = NULL;
>+	}
> 	qed_ops->common->remove(cdev);
> 	edev->cdev = NULL;
> 
>diff --git a/include/linux/qed/qed_if.h b/include/linux/qed/qed_if.h
>index cd6a5c7e56eb..d8368e1770df 100644
>--- a/include/linux/qed/qed_if.h
>+++ b/include/linux/qed/qed_if.h
>@@ -21,6 +21,7 @@
> #include <linux/qed/common_hsi.h>
> #include <linux/qed/qed_chain.h>
> #include <linux/io-64-nonatomic-lo-hi.h>
>+#include <net/devlink.h>
> 
> enum dcbx_protocol_type {
> 	DCBX_PROTOCOL_ISCSI,
>@@ -779,6 +780,10 @@ enum qed_nvm_flash_cmd {
> 	QED_NVM_FLASH_CMD_NVM_MAX,
> };
> 
>+struct qed_devlink {
>+	struct qed_dev *cdev;
>+};
>+
> struct qed_common_cb_ops {
> 	void (*arfs_filter_op)(void *dev, void *fltr, u8 fw_rc);
> 	void (*link_update)(void *dev, struct qed_link_output *link);
>@@ -1137,6 +1142,10 @@ struct qed_common_ops {
>  *
>  */
> 	int (*set_grc_config)(struct qed_dev *cdev, u32 cfg_id, u32 val);
>+
>+	struct devlink* (*devlink_register)(struct qed_dev *cdev);
>+
>+	void (*devlink_unregister)(struct devlink *devlink);
> };
> 
> #define MASK_FIELD(_name, _value) \
>-- 
>2.17.1
>
diff mbox series

Patch

diff --git a/drivers/net/ethernet/qlogic/qed/qed.h b/drivers/net/ethernet/qlogic/qed/qed.h
index b2a7b53ee760..b6ce1488abcc 100644
--- a/drivers/net/ethernet/qlogic/qed/qed.h
+++ b/drivers/net/ethernet/qlogic/qed/qed.h
@@ -849,7 +849,6 @@  struct qed_dev {
 	u32 rdma_max_srq_sge;
 	u16 tunn_feature_mask;
 
-	struct devlink			*dl;
 	bool				iwarp_cmt;
 };
 
diff --git a/drivers/net/ethernet/qlogic/qed/qed_devlink.c b/drivers/net/ethernet/qlogic/qed/qed_devlink.c
index eb693787c99e..a62c47c61edf 100644
--- a/drivers/net/ethernet/qlogic/qed/qed_devlink.c
+++ b/drivers/net/ethernet/qlogic/qed/qed_devlink.c
@@ -5,6 +5,7 @@ 
  */
 
 #include <linux/kernel.h>
+#include <linux/qed/qed_if.h>
 #include "qed.h"
 #include "qed_devlink.h"
 
@@ -13,17 +14,12 @@  enum qed_devlink_param_id {
 	QED_DEVLINK_PARAM_ID_IWARP_CMT,
 };
 
-struct qed_devlink {
-	struct qed_dev *cdev;
-};
-
 static int qed_dl_param_get(struct devlink *dl, u32 id,
 			    struct devlink_param_gset_ctx *ctx)
 {
-	struct qed_devlink *qed_dl;
+	struct qed_devlink *qed_dl = devlink_priv(dl);
 	struct qed_dev *cdev;
 
-	qed_dl = devlink_priv(dl);
 	cdev = qed_dl->cdev;
 	ctx->val.vbool = cdev->iwarp_cmt;
 
@@ -33,10 +29,9 @@  static int qed_dl_param_get(struct devlink *dl, u32 id,
 static int qed_dl_param_set(struct devlink *dl, u32 id,
 			    struct devlink_param_gset_ctx *ctx)
 {
-	struct qed_devlink *qed_dl;
+	struct qed_devlink *qed_dl = devlink_priv(dl);
 	struct qed_dev *cdev;
 
-	qed_dl = devlink_priv(dl);
 	cdev = qed_dl->cdev;
 	cdev->iwarp_cmt = ctx->val.vbool;
 
@@ -52,21 +47,19 @@  static const struct devlink_param qed_devlink_params[] = {
 
 static const struct devlink_ops qed_dl_ops;
 
-int qed_devlink_register(struct qed_dev *cdev)
+struct devlink *qed_devlink_register(struct qed_dev *cdev)
 {
 	union devlink_param_value value;
-	struct qed_devlink *qed_dl;
+	struct qed_devlink *qdevlink;
 	struct devlink *dl;
 	int rc;
 
-	dl = devlink_alloc(&qed_dl_ops, sizeof(*qed_dl));
+	dl = devlink_alloc(&qed_dl_ops, sizeof(struct qed_devlink));
 	if (!dl)
-		return -ENOMEM;
+		return ERR_PTR(-ENOMEM);
 
-	qed_dl = devlink_priv(dl);
-
-	cdev->dl = dl;
-	qed_dl->cdev = cdev;
+	qdevlink = devlink_priv(dl);
+	qdevlink->cdev = cdev;
 
 	rc = devlink_register(dl, &cdev->pdev->dev);
 	if (rc)
@@ -85,26 +78,25 @@  int qed_devlink_register(struct qed_dev *cdev)
 	devlink_params_publish(dl);
 	cdev->iwarp_cmt = false;
 
-	return 0;
+	return dl;
 
 err_unregister:
 	devlink_unregister(dl);
 
 err_free:
-	cdev->dl = NULL;
 	devlink_free(dl);
 
-	return rc;
+	return ERR_PTR(rc);
 }
 
-void qed_devlink_unregister(struct qed_dev *cdev)
+void qed_devlink_unregister(struct devlink *devlink)
 {
-	if (!cdev->dl)
+	if (!devlink)
 		return;
 
-	devlink_params_unregister(cdev->dl, qed_devlink_params,
+	devlink_params_unregister(devlink, qed_devlink_params,
 				  ARRAY_SIZE(qed_devlink_params));
 
-	devlink_unregister(cdev->dl);
-	devlink_free(cdev->dl);
+	devlink_unregister(devlink);
+	devlink_free(devlink);
 }
diff --git a/drivers/net/ethernet/qlogic/qed/qed_devlink.h b/drivers/net/ethernet/qlogic/qed/qed_devlink.h
index b94c40e9b7c1..c79dc6bfa194 100644
--- a/drivers/net/ethernet/qlogic/qed/qed_devlink.h
+++ b/drivers/net/ethernet/qlogic/qed/qed_devlink.h
@@ -9,7 +9,7 @@ 
 #include <linux/qed/qed_if.h>
 #include <net/devlink.h>
 
-int qed_devlink_register(struct qed_dev *cdev);
-void qed_devlink_unregister(struct qed_dev *cdev);
+struct devlink *qed_devlink_register(struct qed_dev *cdev);
+void qed_devlink_unregister(struct devlink *devlink);
 
 #endif
diff --git a/drivers/net/ethernet/qlogic/qed/qed_main.c b/drivers/net/ethernet/qlogic/qed/qed_main.c
index 8751355d9ef7..d6f76421379b 100644
--- a/drivers/net/ethernet/qlogic/qed/qed_main.c
+++ b/drivers/net/ethernet/qlogic/qed/qed_main.c
@@ -539,12 +539,6 @@  static struct qed_dev *qed_probe(struct pci_dev *pdev,
 	}
 	DP_INFO(cdev, "PCI init completed successfully\n");
 
-	rc = qed_devlink_register(cdev);
-	if (rc) {
-		DP_INFO(cdev, "Failed to register devlink.\n");
-		goto err2;
-	}
-
 	rc = qed_hw_prepare(cdev, QED_PCI_DEFAULT);
 	if (rc) {
 		DP_ERR(cdev, "hw prepare failed\n");
@@ -574,8 +568,6 @@  static void qed_remove(struct qed_dev *cdev)
 
 	qed_set_power_state(cdev, PCI_D3hot);
 
-	qed_devlink_unregister(cdev);
-
 	qed_free_cdev(cdev);
 }
 
@@ -3012,6 +3004,8 @@  const struct qed_common_ops qed_common_ops_pass = {
 	.get_link = &qed_get_current_link,
 	.drain = &qed_drain,
 	.update_msglvl = &qed_init_dp,
+	.devlink_register = qed_devlink_register,
+	.devlink_unregister = qed_devlink_unregister,
 	.dbg_all_data = &qed_dbg_all_data,
 	.dbg_all_data_size = &qed_dbg_all_data_size,
 	.chain_alloc = &qed_chain_alloc,
diff --git a/drivers/net/ethernet/qlogic/qede/qede.h b/drivers/net/ethernet/qlogic/qede/qede.h
index 803c1fcca8ad..1f0e7505a973 100644
--- a/drivers/net/ethernet/qlogic/qede/qede.h
+++ b/drivers/net/ethernet/qlogic/qede/qede.h
@@ -172,6 +172,7 @@  struct qede_dev {
 	struct qed_dev			*cdev;
 	struct net_device		*ndev;
 	struct pci_dev			*pdev;
+	struct devlink			*devlink;
 
 	u32				dp_module;
 	u8				dp_level;
diff --git a/drivers/net/ethernet/qlogic/qede/qede_main.c b/drivers/net/ethernet/qlogic/qede/qede_main.c
index 1aaae3203f5a..7c2d948b2035 100644
--- a/drivers/net/ethernet/qlogic/qede/qede_main.c
+++ b/drivers/net/ethernet/qlogic/qede/qede_main.c
@@ -1172,10 +1172,23 @@  static int __qede_probe(struct pci_dev *pdev, u32 dp_module, u8 dp_level,
 			rc = -ENOMEM;
 			goto err2;
 		}
+
+		edev->devlink = qed_ops->common->devlink_register(cdev);
+		if (IS_ERR(edev->devlink)) {
+			DP_NOTICE(edev, "Cannot register devlink\n");
+			edev->devlink = NULL;
+			/* Go on, we can live without devlink */
+		}
 	} else {
 		struct net_device *ndev = pci_get_drvdata(pdev);
 
 		edev = netdev_priv(ndev);
+
+		if (edev && edev->devlink) {
+			struct qed_devlink *qdl = devlink_priv(edev->devlink);
+
+			qdl->cdev = cdev;
+		}
 		edev->cdev = cdev;
 		memset(&edev->stats, 0, sizeof(edev->stats));
 		memcpy(&edev->dev_info, &dev_info, sizeof(dev_info));
@@ -1298,6 +1311,11 @@  static void __qede_remove(struct pci_dev *pdev, enum qede_remove_mode mode)
 	qed_ops->common->slowpath_stop(cdev);
 	if (system_state == SYSTEM_POWER_OFF)
 		return;
+
+	if (mode != QEDE_REMOVE_RECOVERY && edev->devlink) {
+		qed_ops->common->devlink_unregister(edev->devlink);
+		edev->devlink = NULL;
+	}
 	qed_ops->common->remove(cdev);
 	edev->cdev = NULL;
 
diff --git a/include/linux/qed/qed_if.h b/include/linux/qed/qed_if.h
index cd6a5c7e56eb..d8368e1770df 100644
--- a/include/linux/qed/qed_if.h
+++ b/include/linux/qed/qed_if.h
@@ -21,6 +21,7 @@ 
 #include <linux/qed/common_hsi.h>
 #include <linux/qed/qed_chain.h>
 #include <linux/io-64-nonatomic-lo-hi.h>
+#include <net/devlink.h>
 
 enum dcbx_protocol_type {
 	DCBX_PROTOCOL_ISCSI,
@@ -779,6 +780,10 @@  enum qed_nvm_flash_cmd {
 	QED_NVM_FLASH_CMD_NVM_MAX,
 };
 
+struct qed_devlink {
+	struct qed_dev *cdev;
+};
+
 struct qed_common_cb_ops {
 	void (*arfs_filter_op)(void *dev, void *fltr, u8 fw_rc);
 	void (*link_update)(void *dev, struct qed_link_output *link);
@@ -1137,6 +1142,10 @@  struct qed_common_ops {
  *
  */
 	int (*set_grc_config)(struct qed_dev *cdev, u32 cfg_id, u32 val);
+
+	struct devlink* (*devlink_register)(struct qed_dev *cdev);
+
+	void (*devlink_unregister)(struct devlink *devlink);
 };
 
 #define MASK_FIELD(_name, _value) \