sonic-buildimage/platform/mellanox/non-upstream-patches/patches/0053-platform-mellanox-Add-support-for-dynamic-I2C-channe.patch

160 lines
5.5 KiB
Diff
Raw Normal View History

From 7831c5d10460c08cb5837827784677e0286d14e7 Mon Sep 17 00:00:00 2001
From: Vadim Pasternak <vadimp@nvidia.com>
Date: Sun, 30 Jul 2023 21:31:54 +0000
Subject: [PATH backport v6.1 53/54] platform/mellanox: Add support for dynamic
I2C channels infrastructure
Allow to support platform configuration for dynamically allocated I2C
channels.
Motivation is to support I2C channels allocated in a non-continuous
way.
Currently hotplug platform driver data structure contains static mux
channels for I2C hotplug devices. These channels numbers can be updated
dynamically and returned by mux driver's callback through the adapters
array.
Thus, hotplug mux channels will be aligned according to the dynamic
adapters data.
Signed-off-by: Vadim Pasternak <vadimp@nvidia.com>
Reviewed-by: Michael Shych <michaelsh@nvidia.com>
---
drivers/platform/x86/mlx-platform.c | 69 ++++++++++++++++++++++++-----
1 file changed, 59 insertions(+), 10 deletions(-)
diff --git a/drivers/platform/x86/mlx-platform.c b/drivers/platform/x86/mlx-platform.c
index 44f107965832..9021597b5446 100644
--- a/drivers/platform/x86/mlx-platform.c
+++ b/drivers/platform/x86/mlx-platform.c
@@ -15,6 +15,7 @@
#include <linux/pci.h>
#include <linux/platform_device.h>
#include <linux/platform_data/i2c-mux-reg.h>
+#include <linux/platform_data/i2c-mux-regmap.h>
#include <linux/platform_data/mlxreg.h>
#include <linux/reboot.h>
#include <linux/regmap.h>
@@ -350,6 +351,7 @@
* @hotplug_resources: system hotplug resources
* @hotplug_resources_size: size of system hotplug resources
* @hi2c_main_init_status: init status of I2C main bus
+ * @mux_added: number of added mux segments
* @irq_fpga: FPGA IRQ number
*/
struct mlxplat_priv {
@@ -364,6 +366,7 @@ struct mlxplat_priv {
struct resource *hotplug_resources;
unsigned int hotplug_resources_size;
u8 i2c_main_init_status;
+ int mux_added;
int irq_fpga;
};
@@ -453,7 +456,9 @@ static struct i2c_mux_reg_platform_data mlxplat_default_mux_data[] = {
/* Platform mux configuration variables */
static int mlxplat_max_adap_num;
static int mlxplat_mux_num;
+static int mlxplat_mux_hotplug_num;
static struct i2c_mux_reg_platform_data *mlxplat_mux_data;
+static struct i2c_mux_regmap_platform_data *mlxplat_mux_regmap_data;
static struct notifier_block *mlxplat_reboot_nb;
/* Platform extended mux data */
@@ -6126,12 +6131,17 @@ static int mlxplat_mlxcpld_verify_bus_topology(int *nr)
/* Shift adapter ids, since expected parent adapter is not free. */
*nr = i;
for (i = 0; i < mlxplat_mux_num; i++) {
- shift = *nr - mlxplat_mux_data[i].parent;
- mlxplat_mux_data[i].parent = *nr;
- mlxplat_mux_data[i].base_nr += shift;
+ if (mlxplat_mux_data) {
+ shift = *nr - mlxplat_mux_data[i].parent;
+ mlxplat_mux_data[i].parent = *nr;
+ mlxplat_mux_data[i].base_nr += shift;
+ } else if (mlxplat_mux_regmap_data) {
+ mlxplat_mux_regmap_data[i].parent = *nr;
+ }
}
- if (shift > 0)
+ /* Shift bus only if mux provided by 'mlxplat_mux_data'. */
+ if (shift > 0 && mlxplat_mux_data)
mlxplat_hotplug->shift_nr = shift;
return 0;
@@ -6428,8 +6438,31 @@ mlxplat_i2c_mux_complition_notify(void *handle, struct i2c_adapter *parent,
struct i2c_adapter *adapters[])
{
struct mlxplat_priv *priv = handle;
+ struct mlxreg_core_item *item;
+ int i, j;
+
+ /*
+ * Hotplug platform driver data structure contains static mux channels for I2C hotplug
+ * devices. These channels numbers can be updated dynamically and returned by mux callback
+ * through the adapters array. Update mux channels according to the dynamic adapters data.
+ */
+ if (priv->mux_added == mlxplat_mux_hotplug_num) {
+ item = mlxplat_hotplug->items;
+ for (i = 0; i < mlxplat_hotplug->counter; i++, item++) {
+ struct mlxreg_core_data *data = item->data;
+
+ for (j = 0; j < item->count; j++, data++) {
+ if (data->hpdev.nr != MLXPLAT_CPLD_NR_NONE)
+ data->hpdev.nr = adapters[data->hpdev.nr - 2]->nr;
+ }
+ }
+ }
- return mlxplat_post_init(priv);
+ /* Start post initialization only after last nux segment is added */
+ if (++priv->mux_added == mlxplat_mux_num)
+ return mlxplat_post_init(priv);
+
+ return 0;
}
static int mlxplat_i2c_mux_topology_init(struct mlxplat_priv *priv)
@@ -6443,17 +6476,33 @@ static int mlxplat_i2c_mux_topology_init(struct mlxplat_priv *priv)
priv->i2c_main_init_status = MLXPLAT_I2C_MAIN_BUS_HANDLE_CREATED;
for (i = 0; i < mlxplat_mux_num; i++) {
- priv->pdev_mux[i] = platform_device_register_resndata(&priv->pdev_i2c->dev,
- "i2c-mux-reg", i, NULL, 0,
- &mlxplat_mux_data[i],
- sizeof(mlxplat_mux_data[i]));
+ if (mlxplat_mux_data) {
+ priv->pdev_mux[i] =
+ platform_device_register_resndata(&priv->pdev_i2c->dev,
+ "i2c-mux-reg", i, NULL, 0,
+ &mlxplat_mux_data[i],
+ sizeof(mlxplat_mux_data[i]));
+ } else {
+ mlxplat_mux_regmap_data[i].handle = priv;
+ mlxplat_mux_regmap_data[i].regmap = priv->regmap;
+ mlxplat_mux_regmap_data[i].completion_notify =
+ mlxplat_i2c_mux_complition_notify;
+ priv->pdev_mux[i] =
+ platform_device_register_resndata(&priv->pdev_i2c->dev,
+ "i2c-mux-regmap", i, NULL, 0,
+ &mlxplat_mux_regmap_data[i],
+ sizeof(mlxplat_mux_regmap_data[i]));
+ }
if (IS_ERR(priv->pdev_mux[i])) {
err = PTR_ERR(priv->pdev_mux[i]);
goto fail_platform_mux_register;
}
}
- return mlxplat_i2c_mux_complition_notify(priv, NULL, NULL);
+ if (mlxplat_mux_regmap_data && mlxplat_mux_regmap_data->completion_notify)
+ return 0;
+
+ return mlxplat_post_init(priv);
fail_platform_mux_register:
while (--i >= 0)
--
2.20.1