diff options
Diffstat (limited to 'drivers/soc/imx/imx8m-blk-ctrl.c')
| -rw-r--r-- | drivers/soc/imx/imx8m-blk-ctrl.c | 96 | 
1 files changed, 95 insertions, 1 deletions
| diff --git a/drivers/soc/imx/imx8m-blk-ctrl.c b/drivers/soc/imx/imx8m-blk-ctrl.c index 519b3651d1d9..511e74f0db8a 100644 --- a/drivers/soc/imx/imx8m-blk-ctrl.c +++ b/drivers/soc/imx/imx8m-blk-ctrl.c @@ -14,9 +14,11 @@  #include <linux/clk.h>  #include <dt-bindings/power/imx8mm-power.h> +#include <dt-bindings/power/imx8mn-power.h>  #define BLK_SFT_RSTN	0x0  #define BLK_CLK_EN	0x4 +#define BLK_MIPI_RESET_DIV	0x8 /* Mini/Nano DISPLAY_BLK_CTRL only */  struct imx8m_blk_ctrl_domain; @@ -36,6 +38,15 @@ struct imx8m_blk_ctrl_domain_data {  	const char *gpc_name;  	u32 rst_mask;  	u32 clk_mask; + +	/* +	 * i.MX8M Mini and Nano have a third DISPLAY_BLK_CTRL register +	 * which is used to control the reset for the MIPI Phy. +	 * Since it's only present in certain circumstances, +	 * an if-statement should be used before setting and clearing this +	 * register. +	 */ +	u32 mipi_phy_rst_mask;  };  #define DOMAIN_MAX_CLKS 3 @@ -78,6 +89,8 @@ static int imx8m_blk_ctrl_power_on(struct generic_pm_domain *genpd)  	/* put devices into reset */  	regmap_clear_bits(bc->regmap, BLK_SFT_RSTN, data->rst_mask); +	if (data->mipi_phy_rst_mask) +		regmap_clear_bits(bc->regmap, BLK_MIPI_RESET_DIV, data->mipi_phy_rst_mask);  	/* enable upstream and blk-ctrl clocks to allow reset to propagate */  	ret = clk_bulk_prepare_enable(data->num_clks, domain->clks); @@ -99,6 +112,8 @@ static int imx8m_blk_ctrl_power_on(struct generic_pm_domain *genpd)  	/* release reset */  	regmap_set_bits(bc->regmap, BLK_SFT_RSTN, data->rst_mask); +	if (data->mipi_phy_rst_mask) +		regmap_set_bits(bc->regmap, BLK_MIPI_RESET_DIV, data->mipi_phy_rst_mask);  	/* disable upstream clocks */  	clk_bulk_disable_unprepare(data->num_clks, domain->clks); @@ -120,6 +135,9 @@ static int imx8m_blk_ctrl_power_off(struct generic_pm_domain *genpd)  	struct imx8m_blk_ctrl *bc = domain->bc;  	/* put devices into reset and disable clocks */ +	if (data->mipi_phy_rst_mask) +		regmap_clear_bits(bc->regmap, BLK_MIPI_RESET_DIV, data->mipi_phy_rst_mask); +  	regmap_clear_bits(bc->regmap, BLK_SFT_RSTN, data->rst_mask);  	regmap_clear_bits(bc->regmap, BLK_CLK_EN, data->clk_mask); @@ -480,6 +498,7 @@ static const struct imx8m_blk_ctrl_domain_data imx8mm_disp_blk_ctl_domain_data[]  		.gpc_name = "mipi-dsi",  		.rst_mask = BIT(5),  		.clk_mask = BIT(8) | BIT(9), +		.mipi_phy_rst_mask = BIT(17),  	},  	[IMX8MM_DISPBLK_PD_MIPI_CSI] = {  		.name = "dispblk-mipi-csi", @@ -488,6 +507,7 @@ static const struct imx8m_blk_ctrl_domain_data imx8mm_disp_blk_ctl_domain_data[]  		.gpc_name = "mipi-csi",  		.rst_mask = BIT(3) | BIT(4),  		.clk_mask = BIT(10) | BIT(11), +		.mipi_phy_rst_mask = BIT(16),  	},  }; @@ -498,6 +518,77 @@ static const struct imx8m_blk_ctrl_data imx8mm_disp_blk_ctl_dev_data = {  	.num_domains = ARRAY_SIZE(imx8mm_disp_blk_ctl_domain_data),  }; + +static int imx8mn_disp_power_notifier(struct notifier_block *nb, +				      unsigned long action, void *data) +{ +	struct imx8m_blk_ctrl *bc = container_of(nb, struct imx8m_blk_ctrl, +						 power_nb); + +	if (action != GENPD_NOTIFY_ON && action != GENPD_NOTIFY_PRE_OFF) +		return NOTIFY_OK; + +	/* Enable bus clock and deassert bus reset */ +	regmap_set_bits(bc->regmap, BLK_CLK_EN, BIT(8)); +	regmap_set_bits(bc->regmap, BLK_SFT_RSTN, BIT(8)); + +	/* +	 * On power up we have no software backchannel to the GPC to +	 * wait for the ADB handshake to happen, so we just delay for a +	 * bit. On power down the GPC driver waits for the handshake. +	 */ +	if (action == GENPD_NOTIFY_ON) +		udelay(5); + + +	return NOTIFY_OK; +} + +static const struct imx8m_blk_ctrl_domain_data imx8mn_disp_blk_ctl_domain_data[] = { +	[IMX8MN_DISPBLK_PD_MIPI_DSI] = { +		.name = "dispblk-mipi-dsi", +		.clk_names = (const char *[]){ "dsi-pclk", "dsi-ref", }, +		.num_clks = 2, +		.gpc_name = "mipi-dsi", +		.rst_mask = BIT(0) | BIT(1), +		.clk_mask = BIT(0) | BIT(1), +		.mipi_phy_rst_mask = BIT(17), +	}, +	[IMX8MN_DISPBLK_PD_MIPI_CSI] = { +		.name = "dispblk-mipi-csi", +		.clk_names = (const char *[]){ "csi-aclk", "csi-pclk" }, +		.num_clks = 2, +		.gpc_name = "mipi-csi", +		.rst_mask = BIT(2) | BIT(3), +		.clk_mask = BIT(2) | BIT(3), +		.mipi_phy_rst_mask = BIT(16), +	}, +	[IMX8MN_DISPBLK_PD_LCDIF] = { +		.name = "dispblk-lcdif", +		.clk_names = (const char *[]){ "lcdif-axi", "lcdif-apb", "lcdif-pix", }, +		.num_clks = 3, +		.gpc_name = "lcdif", +		.rst_mask = BIT(4) | BIT(5), +		.clk_mask = BIT(4) | BIT(5), +	}, +	[IMX8MN_DISPBLK_PD_ISI] = { +		.name = "dispblk-isi", +		.clk_names = (const char *[]){ "disp_axi", "disp_apb", "disp_axi_root", +						"disp_apb_root"}, +		.num_clks = 4, +		.gpc_name = "isi", +		.rst_mask = BIT(6) | BIT(7), +		.clk_mask = BIT(6) | BIT(7), +	}, +}; + +static const struct imx8m_blk_ctrl_data imx8mn_disp_blk_ctl_dev_data = { +	.max_reg = 0x84, +	.power_notifier_fn = imx8mn_disp_power_notifier, +	.domains = imx8mn_disp_blk_ctl_domain_data, +	.num_domains = ARRAY_SIZE(imx8mn_disp_blk_ctl_domain_data), +}; +  static const struct of_device_id imx8m_blk_ctrl_of_match[] = {  	{  		.compatible = "fsl,imx8mm-vpu-blk-ctrl", @@ -505,7 +596,10 @@ static const struct of_device_id imx8m_blk_ctrl_of_match[] = {  	}, {  		.compatible = "fsl,imx8mm-disp-blk-ctrl",  		.data = &imx8mm_disp_blk_ctl_dev_data -	} ,{ +	}, { +		.compatible = "fsl,imx8mn-disp-blk-ctrl", +		.data = &imx8mn_disp_blk_ctl_dev_data +	}, {  		/* Sentinel */  	}  }; |