xref: /openbmc/linux/drivers/gpu/drm/bridge/lontium-lt9611.c (revision b181f7029bd71238ac2754ce7052dffd69432085)
123278bf5SVinod Koul // SPDX-License-Identifier: GPL-2.0
223278bf5SVinod Koul /*
323278bf5SVinod Koul  * Copyright (c) 2018, The Linux Foundation. All rights reserved.
423278bf5SVinod Koul  * Copyright (c) 2019-2020. Linaro Limited.
523278bf5SVinod Koul  */
623278bf5SVinod Koul 
723278bf5SVinod Koul #include <linux/gpio/consumer.h>
8a204f974SVille Syrjälä #include <linux/i2c.h>
923278bf5SVinod Koul #include <linux/interrupt.h>
1072bd9ea3SVille Syrjälä #include <linux/media-bus-format.h>
1123278bf5SVinod Koul #include <linux/module.h>
1223278bf5SVinod Koul #include <linux/of_graph.h>
1323278bf5SVinod Koul #include <linux/platform_device.h>
1423278bf5SVinod Koul #include <linux/regmap.h>
1523278bf5SVinod Koul #include <linux/regulator/consumer.h>
1623278bf5SVinod Koul 
1723278bf5SVinod Koul #include <sound/hdmi-codec.h>
1823278bf5SVinod Koul 
1923278bf5SVinod Koul #include <drm/drm_atomic_helper.h>
2023278bf5SVinod Koul #include <drm/drm_bridge.h>
2123278bf5SVinod Koul #include <drm/drm_mipi_dsi.h>
220c3997b0SDmitry Baryshkov #include <drm/drm_of.h>
2323278bf5SVinod Koul #include <drm/drm_print.h>
2423278bf5SVinod Koul #include <drm/drm_probe_helper.h>
2523278bf5SVinod Koul 
2623278bf5SVinod Koul #define EDID_SEG_SIZE	256
2723278bf5SVinod Koul #define EDID_LEN	32
2823278bf5SVinod Koul #define EDID_LOOP	8
2923278bf5SVinod Koul #define KEY_DDC_ACCS_DONE 0x02
3023278bf5SVinod Koul #define DDC_NO_ACK	0x50
3123278bf5SVinod Koul 
3223278bf5SVinod Koul #define LT9611_4LANES	0
3323278bf5SVinod Koul 
3423278bf5SVinod Koul struct lt9611 {
3523278bf5SVinod Koul 	struct device *dev;
3623278bf5SVinod Koul 	struct drm_bridge bridge;
370c3997b0SDmitry Baryshkov 	struct drm_bridge *next_bridge;
3823278bf5SVinod Koul 
3923278bf5SVinod Koul 	struct regmap *regmap;
4023278bf5SVinod Koul 
4123278bf5SVinod Koul 	struct device_node *dsi0_node;
4223278bf5SVinod Koul 	struct device_node *dsi1_node;
4323278bf5SVinod Koul 	struct mipi_dsi_device *dsi0;
4423278bf5SVinod Koul 	struct mipi_dsi_device *dsi1;
4523278bf5SVinod Koul 	struct platform_device *audio_pdev;
4623278bf5SVinod Koul 
4723278bf5SVinod Koul 	bool ac_mode;
4823278bf5SVinod Koul 
4923278bf5SVinod Koul 	struct gpio_desc *reset_gpio;
5023278bf5SVinod Koul 	struct gpio_desc *enable_gpio;
5123278bf5SVinod Koul 
5223278bf5SVinod Koul 	bool power_on;
5323278bf5SVinod Koul 	bool sleep;
5423278bf5SVinod Koul 
5523278bf5SVinod Koul 	struct regulator_bulk_data supplies[2];
5623278bf5SVinod Koul 
5723278bf5SVinod Koul 	struct i2c_client *client;
5823278bf5SVinod Koul 
5923278bf5SVinod Koul 	enum drm_connector_status status;
6023278bf5SVinod Koul 
6123278bf5SVinod Koul 	u8 edid_buf[EDID_SEG_SIZE];
6223278bf5SVinod Koul };
6323278bf5SVinod Koul 
6423278bf5SVinod Koul #define LT9611_PAGE_CONTROL	0xff
6523278bf5SVinod Koul 
6623278bf5SVinod Koul static const struct regmap_range_cfg lt9611_ranges[] = {
6723278bf5SVinod Koul 	{
6823278bf5SVinod Koul 		.name = "register_range",
6923278bf5SVinod Koul 		.range_min =  0,
7023278bf5SVinod Koul 		.range_max = 0x85ff,
7123278bf5SVinod Koul 		.selector_reg = LT9611_PAGE_CONTROL,
7223278bf5SVinod Koul 		.selector_mask = 0xff,
7323278bf5SVinod Koul 		.selector_shift = 0,
7423278bf5SVinod Koul 		.window_start = 0,
7523278bf5SVinod Koul 		.window_len = 0x100,
7623278bf5SVinod Koul 	},
7723278bf5SVinod Koul };
7823278bf5SVinod Koul 
7923278bf5SVinod Koul static const struct regmap_config lt9611_regmap_config = {
8023278bf5SVinod Koul 	.reg_bits = 8,
8123278bf5SVinod Koul 	.val_bits = 8,
8223278bf5SVinod Koul 	.max_register = 0xffff,
8323278bf5SVinod Koul 	.ranges = lt9611_ranges,
8423278bf5SVinod Koul 	.num_ranges = ARRAY_SIZE(lt9611_ranges),
8523278bf5SVinod Koul };
8623278bf5SVinod Koul 
bridge_to_lt9611(struct drm_bridge * bridge)8723278bf5SVinod Koul static struct lt9611 *bridge_to_lt9611(struct drm_bridge *bridge)
8823278bf5SVinod Koul {
8923278bf5SVinod Koul 	return container_of(bridge, struct lt9611, bridge);
9023278bf5SVinod Koul }
9123278bf5SVinod Koul 
lt9611_mipi_input_analog(struct lt9611 * lt9611)9223278bf5SVinod Koul static int lt9611_mipi_input_analog(struct lt9611 *lt9611)
9323278bf5SVinod Koul {
9423278bf5SVinod Koul 	const struct reg_sequence reg_cfg[] = {
9523278bf5SVinod Koul 		{ 0x8106, 0x40 }, /* port A rx current */
9623278bf5SVinod Koul 		{ 0x810a, 0xfe }, /* port A ldo voltage set */
9723278bf5SVinod Koul 		{ 0x810b, 0xbf }, /* enable port A lprx */
9823278bf5SVinod Koul 		{ 0x8111, 0x40 }, /* port B rx current */
9923278bf5SVinod Koul 		{ 0x8115, 0xfe }, /* port B ldo voltage set */
10023278bf5SVinod Koul 		{ 0x8116, 0xbf }, /* enable port B lprx */
10123278bf5SVinod Koul 
10223278bf5SVinod Koul 		{ 0x811c, 0x03 }, /* PortA clk lane no-LP mode */
10323278bf5SVinod Koul 		{ 0x8120, 0x03 }, /* PortB clk lane with-LP mode */
10423278bf5SVinod Koul 	};
10523278bf5SVinod Koul 
10623278bf5SVinod Koul 	return regmap_multi_reg_write(lt9611->regmap, reg_cfg, ARRAY_SIZE(reg_cfg));
10723278bf5SVinod Koul }
10823278bf5SVinod Koul 
lt9611_mipi_input_digital(struct lt9611 * lt9611,const struct drm_display_mode * mode)10923278bf5SVinod Koul static int lt9611_mipi_input_digital(struct lt9611 *lt9611,
11023278bf5SVinod Koul 				     const struct drm_display_mode *mode)
11123278bf5SVinod Koul {
11223278bf5SVinod Koul 	struct reg_sequence reg_cfg[] = {
11323278bf5SVinod Koul 		{ 0x8300, LT9611_4LANES },
11423278bf5SVinod Koul 		{ 0x830a, 0x00 },
11523278bf5SVinod Koul 		{ 0x824f, 0x80 },
11623278bf5SVinod Koul 		{ 0x8250, 0x10 },
11723278bf5SVinod Koul 		{ 0x8302, 0x0a },
11823278bf5SVinod Koul 		{ 0x8306, 0x0a },
11923278bf5SVinod Koul 	};
12023278bf5SVinod Koul 
1215e83f359SDmitry Baryshkov 	if (lt9611->dsi1_node)
12223278bf5SVinod Koul 		reg_cfg[1].def = 0x03;
12323278bf5SVinod Koul 
12423278bf5SVinod Koul 	return regmap_multi_reg_write(lt9611->regmap, reg_cfg, ARRAY_SIZE(reg_cfg));
12523278bf5SVinod Koul }
12623278bf5SVinod Koul 
lt9611_mipi_video_setup(struct lt9611 * lt9611,const struct drm_display_mode * mode)12723278bf5SVinod Koul static void lt9611_mipi_video_setup(struct lt9611 *lt9611,
12823278bf5SVinod Koul 				    const struct drm_display_mode *mode)
12923278bf5SVinod Koul {
13023278bf5SVinod Koul 	u32 h_total, hactive, hsync_len, hfront_porch, hsync_porch;
13123278bf5SVinod Koul 	u32 v_total, vactive, vsync_len, vfront_porch, vsync_porch;
13223278bf5SVinod Koul 
13323278bf5SVinod Koul 	h_total = mode->htotal;
13423278bf5SVinod Koul 	v_total = mode->vtotal;
13523278bf5SVinod Koul 
13623278bf5SVinod Koul 	hactive = mode->hdisplay;
13723278bf5SVinod Koul 	hsync_len = mode->hsync_end - mode->hsync_start;
13823278bf5SVinod Koul 	hfront_porch = mode->hsync_start - mode->hdisplay;
1396b089d5eSDmitry Baryshkov 	hsync_porch = mode->htotal - mode->hsync_start;
14023278bf5SVinod Koul 
14123278bf5SVinod Koul 	vactive = mode->vdisplay;
14223278bf5SVinod Koul 	vsync_len = mode->vsync_end - mode->vsync_start;
14323278bf5SVinod Koul 	vfront_porch = mode->vsync_start - mode->vdisplay;
1446b089d5eSDmitry Baryshkov 	vsync_porch = mode->vtotal - mode->vsync_start;
14523278bf5SVinod Koul 
14623278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x830d, (u8)(v_total / 256));
14723278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x830e, (u8)(v_total % 256));
14823278bf5SVinod Koul 
14923278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x830f, (u8)(vactive / 256));
15023278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8310, (u8)(vactive % 256));
15123278bf5SVinod Koul 
15223278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8311, (u8)(h_total / 256));
15323278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8312, (u8)(h_total % 256));
15423278bf5SVinod Koul 
15523278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8313, (u8)(hactive / 256));
15623278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8314, (u8)(hactive % 256));
15723278bf5SVinod Koul 
15823278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8315, (u8)(vsync_len % 256));
15923278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8316, (u8)(hsync_len % 256));
16023278bf5SVinod Koul 
16123278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8317, (u8)(vfront_porch % 256));
16223278bf5SVinod Koul 
16323278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8318, (u8)(vsync_porch % 256));
16423278bf5SVinod Koul 
16523278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8319, (u8)(hfront_porch % 256));
16623278bf5SVinod Koul 
167ad188aa4SDmitry Baryshkov 	regmap_write(lt9611->regmap, 0x831a, (u8)(hsync_porch / 256) |
168ad188aa4SDmitry Baryshkov 						((hfront_porch / 256) << 4));
16923278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x831b, (u8)(hsync_porch % 256));
17023278bf5SVinod Koul }
17123278bf5SVinod Koul 
lt9611_pcr_setup(struct lt9611 * lt9611,const struct drm_display_mode * mode,unsigned int postdiv)1722576eb26SDmitry Baryshkov static void lt9611_pcr_setup(struct lt9611 *lt9611, const struct drm_display_mode *mode, unsigned int postdiv)
17323278bf5SVinod Koul {
1742576eb26SDmitry Baryshkov 	unsigned int pcr_m = mode->clock * 5 * postdiv / 27000;
17523278bf5SVinod Koul 	const struct reg_sequence reg_cfg[] = {
17623278bf5SVinod Koul 		{ 0x830b, 0x01 },
17723278bf5SVinod Koul 		{ 0x830c, 0x10 },
17823278bf5SVinod Koul 		{ 0x8348, 0x00 },
17923278bf5SVinod Koul 		{ 0x8349, 0x81 },
18023278bf5SVinod Koul 
18123278bf5SVinod Koul 		/* stage 1 */
18223278bf5SVinod Koul 		{ 0x8321, 0x4a },
18323278bf5SVinod Koul 		{ 0x8324, 0x71 },
18423278bf5SVinod Koul 		{ 0x8325, 0x30 },
18523278bf5SVinod Koul 		{ 0x832a, 0x01 },
18623278bf5SVinod Koul 
18723278bf5SVinod Koul 		/* stage 2 */
18823278bf5SVinod Koul 		{ 0x834a, 0x40 },
18923278bf5SVinod Koul 
19023278bf5SVinod Koul 		/* MK limit */
19123278bf5SVinod Koul 		{ 0x832d, 0x38 },
19223278bf5SVinod Koul 		{ 0x8331, 0x08 },
19323278bf5SVinod Koul 	};
1940b157efaSDmitry Baryshkov 	u8 pol = 0x10;
19523278bf5SVinod Koul 
1960b157efaSDmitry Baryshkov 	if (mode->flags & DRM_MODE_FLAG_NHSYNC)
1970b157efaSDmitry Baryshkov 		pol |= 0x2;
1980b157efaSDmitry Baryshkov 	if (mode->flags & DRM_MODE_FLAG_NVSYNC)
1990b157efaSDmitry Baryshkov 		pol |= 0x1;
2000b157efaSDmitry Baryshkov 	regmap_write(lt9611->regmap, 0x831d, pol);
2010b157efaSDmitry Baryshkov 
20223278bf5SVinod Koul 	regmap_multi_reg_write(lt9611->regmap, reg_cfg, ARRAY_SIZE(reg_cfg));
2035e83f359SDmitry Baryshkov 	if (lt9611->dsi1_node) {
2045e83f359SDmitry Baryshkov 		unsigned int hact = mode->hdisplay;
2055e83f359SDmitry Baryshkov 
2065e83f359SDmitry Baryshkov 		hact >>= 2;
2075e83f359SDmitry Baryshkov 		hact += 0x50;
2085e83f359SDmitry Baryshkov 		hact = min(hact, 0x3e0U);
2095e83f359SDmitry Baryshkov 		regmap_write(lt9611->regmap, 0x830b, hact / 256);
2105e83f359SDmitry Baryshkov 		regmap_write(lt9611->regmap, 0x830c, hact % 256);
2115e83f359SDmitry Baryshkov 		regmap_write(lt9611->regmap, 0x8348, hact / 256);
2125e83f359SDmitry Baryshkov 		regmap_write(lt9611->regmap, 0x8349, hact % 256);
2135e83f359SDmitry Baryshkov 	}
21423278bf5SVinod Koul 
2152576eb26SDmitry Baryshkov 	regmap_write(lt9611->regmap, 0x8326, pcr_m);
21623278bf5SVinod Koul 
21723278bf5SVinod Koul 	/* pcr rst */
21823278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8011, 0x5a);
21923278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8011, 0xfa);
22023278bf5SVinod Koul }
22123278bf5SVinod Koul 
lt9611_pll_setup(struct lt9611 * lt9611,const struct drm_display_mode * mode,unsigned int * postdiv)2222576eb26SDmitry Baryshkov static int lt9611_pll_setup(struct lt9611 *lt9611, const struct drm_display_mode *mode, unsigned int *postdiv)
22323278bf5SVinod Koul {
22423278bf5SVinod Koul 	unsigned int pclk = mode->clock;
22523278bf5SVinod Koul 	const struct reg_sequence reg_cfg[] = {
22623278bf5SVinod Koul 		/* txpll init */
22723278bf5SVinod Koul 		{ 0x8123, 0x40 },
22823278bf5SVinod Koul 		{ 0x8124, 0x64 },
22923278bf5SVinod Koul 		{ 0x8125, 0x80 },
23023278bf5SVinod Koul 		{ 0x8126, 0x55 },
23123278bf5SVinod Koul 		{ 0x812c, 0x37 },
23223278bf5SVinod Koul 		{ 0x812f, 0x01 },
23323278bf5SVinod Koul 		{ 0x8126, 0x55 },
23423278bf5SVinod Koul 		{ 0x8127, 0x66 },
23523278bf5SVinod Koul 		{ 0x8128, 0x88 },
2362a9df204SRobert Foss 		{ 0x812a, 0x20 },
23723278bf5SVinod Koul 	};
23823278bf5SVinod Koul 
23923278bf5SVinod Koul 	regmap_multi_reg_write(lt9611->regmap, reg_cfg, ARRAY_SIZE(reg_cfg));
24023278bf5SVinod Koul 
2412576eb26SDmitry Baryshkov 	if (pclk > 150000) {
24223278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x812d, 0x88);
2432576eb26SDmitry Baryshkov 		*postdiv = 1;
2442576eb26SDmitry Baryshkov 	} else if (pclk > 70000) {
24523278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x812d, 0x99);
2462576eb26SDmitry Baryshkov 		*postdiv = 2;
2472576eb26SDmitry Baryshkov 	} else {
24823278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x812d, 0xaa);
2492576eb26SDmitry Baryshkov 		*postdiv = 4;
2502576eb26SDmitry Baryshkov 	}
25123278bf5SVinod Koul 
25223278bf5SVinod Koul 	/*
25323278bf5SVinod Koul 	 * first divide pclk by 2 first
25423278bf5SVinod Koul 	 *  - write divide by 64k to 19:16 bits which means shift by 17
25523278bf5SVinod Koul 	 *  - write divide by 256 to 15:8 bits which means shift by 9
25623278bf5SVinod Koul 	 *  - write remainder to 7:0 bits, which means shift by 1
25723278bf5SVinod Koul 	 */
25823278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x82e3, pclk >> 17); /* pclk[19:16] */
25923278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x82e4, pclk >> 9);  /* pclk[15:8]  */
26023278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x82e5, pclk >> 1);  /* pclk[7:0]   */
26123278bf5SVinod Koul 
26223278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x82de, 0x20);
26323278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x82de, 0xe0);
26423278bf5SVinod Koul 
26523278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8016, 0xf1);
26623278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8016, 0xf3);
26723278bf5SVinod Koul 
26823278bf5SVinod Koul 	return 0;
26923278bf5SVinod Koul }
27023278bf5SVinod Koul 
lt9611_read_video_check(struct lt9611 * lt9611,unsigned int reg)27123278bf5SVinod Koul static int lt9611_read_video_check(struct lt9611 *lt9611, unsigned int reg)
27223278bf5SVinod Koul {
27323278bf5SVinod Koul 	unsigned int temp, temp2;
27423278bf5SVinod Koul 	int ret;
27523278bf5SVinod Koul 
27623278bf5SVinod Koul 	ret = regmap_read(lt9611->regmap, reg, &temp);
27723278bf5SVinod Koul 	if (ret)
27823278bf5SVinod Koul 		return ret;
27923278bf5SVinod Koul 	temp <<= 8;
28023278bf5SVinod Koul 	ret = regmap_read(lt9611->regmap, reg + 1, &temp2);
28123278bf5SVinod Koul 	if (ret)
28223278bf5SVinod Koul 		return ret;
28323278bf5SVinod Koul 
28423278bf5SVinod Koul 	return (temp + temp2);
28523278bf5SVinod Koul }
28623278bf5SVinod Koul 
lt9611_video_check(struct lt9611 * lt9611)28723278bf5SVinod Koul static int lt9611_video_check(struct lt9611 *lt9611)
28823278bf5SVinod Koul {
28923278bf5SVinod Koul 	u32 v_total, vactive, hactive_a, hactive_b, h_total_sysclk;
29023278bf5SVinod Koul 	int temp;
29123278bf5SVinod Koul 
29223278bf5SVinod Koul 	/* top module video check */
29323278bf5SVinod Koul 
29423278bf5SVinod Koul 	/* vactive */
29523278bf5SVinod Koul 	temp = lt9611_read_video_check(lt9611, 0x8282);
29623278bf5SVinod Koul 	if (temp < 0)
29723278bf5SVinod Koul 		goto end;
29823278bf5SVinod Koul 	vactive = temp;
29923278bf5SVinod Koul 
30023278bf5SVinod Koul 	/* v_total */
30123278bf5SVinod Koul 	temp = lt9611_read_video_check(lt9611, 0x826c);
30223278bf5SVinod Koul 	if (temp < 0)
30323278bf5SVinod Koul 		goto end;
30423278bf5SVinod Koul 	v_total = temp;
30523278bf5SVinod Koul 
30623278bf5SVinod Koul 	/* h_total_sysclk */
30723278bf5SVinod Koul 	temp = lt9611_read_video_check(lt9611, 0x8286);
30823278bf5SVinod Koul 	if (temp < 0)
30923278bf5SVinod Koul 		goto end;
31023278bf5SVinod Koul 	h_total_sysclk = temp;
31123278bf5SVinod Koul 
31223278bf5SVinod Koul 	/* hactive_a */
31323278bf5SVinod Koul 	temp = lt9611_read_video_check(lt9611, 0x8382);
31423278bf5SVinod Koul 	if (temp < 0)
31523278bf5SVinod Koul 		goto end;
31623278bf5SVinod Koul 	hactive_a = temp / 3;
31723278bf5SVinod Koul 
31823278bf5SVinod Koul 	/* hactive_b */
31923278bf5SVinod Koul 	temp = lt9611_read_video_check(lt9611, 0x8386);
32023278bf5SVinod Koul 	if (temp < 0)
32123278bf5SVinod Koul 		goto end;
32223278bf5SVinod Koul 	hactive_b = temp / 3;
32323278bf5SVinod Koul 
32423278bf5SVinod Koul 	dev_info(lt9611->dev,
32523278bf5SVinod Koul 		 "video check: hactive_a=%d, hactive_b=%d, vactive=%d, v_total=%d, h_total_sysclk=%d\n",
32623278bf5SVinod Koul 		 hactive_a, hactive_b, vactive, v_total, h_total_sysclk);
32723278bf5SVinod Koul 
32823278bf5SVinod Koul 	return 0;
32923278bf5SVinod Koul 
33023278bf5SVinod Koul end:
33123278bf5SVinod Koul 	dev_err(lt9611->dev, "read video check error\n");
33223278bf5SVinod Koul 	return temp;
33323278bf5SVinod Koul }
33423278bf5SVinod Koul 
lt9611_hdmi_set_infoframes(struct lt9611 * lt9611,struct drm_connector * connector,struct drm_display_mode * mode)33584cf74d9SDmitry Baryshkov static void lt9611_hdmi_set_infoframes(struct lt9611 *lt9611,
33684cf74d9SDmitry Baryshkov 				       struct drm_connector *connector,
33784cf74d9SDmitry Baryshkov 				       struct drm_display_mode *mode)
33884cf74d9SDmitry Baryshkov {
33984cf74d9SDmitry Baryshkov 	union hdmi_infoframe infoframe;
34084cf74d9SDmitry Baryshkov 	ssize_t len;
34184cf74d9SDmitry Baryshkov 	u8 iframes = 0x0a; /* UD1 infoframe */
34284cf74d9SDmitry Baryshkov 	u8 buf[32];
34384cf74d9SDmitry Baryshkov 	int ret;
34484cf74d9SDmitry Baryshkov 	int i;
34584cf74d9SDmitry Baryshkov 
34684cf74d9SDmitry Baryshkov 	ret = drm_hdmi_avi_infoframe_from_display_mode(&infoframe.avi,
34784cf74d9SDmitry Baryshkov 						       connector,
34884cf74d9SDmitry Baryshkov 						       mode);
34984cf74d9SDmitry Baryshkov 	if (ret < 0)
35084cf74d9SDmitry Baryshkov 		goto out;
35184cf74d9SDmitry Baryshkov 
35284cf74d9SDmitry Baryshkov 	len = hdmi_infoframe_pack(&infoframe, buf, sizeof(buf));
35384cf74d9SDmitry Baryshkov 	if (len < 0)
35484cf74d9SDmitry Baryshkov 		goto out;
35584cf74d9SDmitry Baryshkov 
35684cf74d9SDmitry Baryshkov 	for (i = 0; i < len; i++)
35784cf74d9SDmitry Baryshkov 		regmap_write(lt9611->regmap, 0x8440 + i, buf[i]);
35884cf74d9SDmitry Baryshkov 
35984cf74d9SDmitry Baryshkov 	ret = drm_hdmi_vendor_infoframe_from_display_mode(&infoframe.vendor.hdmi,
36084cf74d9SDmitry Baryshkov 							  connector,
36184cf74d9SDmitry Baryshkov 							  mode);
36284cf74d9SDmitry Baryshkov 	if (ret < 0)
36384cf74d9SDmitry Baryshkov 		goto out;
36484cf74d9SDmitry Baryshkov 
36584cf74d9SDmitry Baryshkov 	len = hdmi_infoframe_pack(&infoframe, buf, sizeof(buf));
36684cf74d9SDmitry Baryshkov 	if (len < 0)
36784cf74d9SDmitry Baryshkov 		goto out;
36884cf74d9SDmitry Baryshkov 
36984cf74d9SDmitry Baryshkov 	for (i = 0; i < len; i++)
37084cf74d9SDmitry Baryshkov 		regmap_write(lt9611->regmap, 0x8474 + i, buf[i]);
37184cf74d9SDmitry Baryshkov 
37284cf74d9SDmitry Baryshkov 	iframes |= 0x20;
37384cf74d9SDmitry Baryshkov 
37484cf74d9SDmitry Baryshkov out:
37584cf74d9SDmitry Baryshkov 	regmap_write(lt9611->regmap, 0x843d, iframes); /* UD1 infoframe */
37684cf74d9SDmitry Baryshkov }
37784cf74d9SDmitry Baryshkov 
lt9611_hdmi_tx_digital(struct lt9611 * lt9611,bool is_hdmi)3780c747469SDmitry Baryshkov static void lt9611_hdmi_tx_digital(struct lt9611 *lt9611, bool is_hdmi)
37923278bf5SVinod Koul {
3800c747469SDmitry Baryshkov 	if (is_hdmi)
38123278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x82d6, 0x8c);
3820c747469SDmitry Baryshkov 	else
3830c747469SDmitry Baryshkov 		regmap_write(lt9611->regmap, 0x82d6, 0x0c);
38423278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x82d7, 0x04);
38523278bf5SVinod Koul }
38623278bf5SVinod Koul 
lt9611_hdmi_tx_phy(struct lt9611 * lt9611)38723278bf5SVinod Koul static void lt9611_hdmi_tx_phy(struct lt9611 *lt9611)
38823278bf5SVinod Koul {
38923278bf5SVinod Koul 	struct reg_sequence reg_cfg[] = {
39023278bf5SVinod Koul 		{ 0x8130, 0x6a },
39123278bf5SVinod Koul 		{ 0x8131, 0x44 }, /* HDMI DC mode */
39223278bf5SVinod Koul 		{ 0x8132, 0x4a },
39323278bf5SVinod Koul 		{ 0x8133, 0x0b },
39423278bf5SVinod Koul 		{ 0x8134, 0x00 },
39523278bf5SVinod Koul 		{ 0x8135, 0x00 },
39623278bf5SVinod Koul 		{ 0x8136, 0x00 },
39723278bf5SVinod Koul 		{ 0x8137, 0x44 },
39823278bf5SVinod Koul 		{ 0x813f, 0x0f },
39923278bf5SVinod Koul 		{ 0x8140, 0xa0 },
40023278bf5SVinod Koul 		{ 0x8141, 0xa0 },
40123278bf5SVinod Koul 		{ 0x8142, 0xa0 },
40223278bf5SVinod Koul 		{ 0x8143, 0xa0 },
40323278bf5SVinod Koul 		{ 0x8144, 0x0a },
40423278bf5SVinod Koul 	};
40523278bf5SVinod Koul 
40623278bf5SVinod Koul 	/* HDMI AC mode */
40723278bf5SVinod Koul 	if (lt9611->ac_mode)
40823278bf5SVinod Koul 		reg_cfg[2].def = 0x73;
40923278bf5SVinod Koul 
41023278bf5SVinod Koul 	regmap_multi_reg_write(lt9611->regmap, reg_cfg, ARRAY_SIZE(reg_cfg));
41123278bf5SVinod Koul }
41223278bf5SVinod Koul 
lt9611_irq_thread_handler(int irq,void * dev_id)41323278bf5SVinod Koul static irqreturn_t lt9611_irq_thread_handler(int irq, void *dev_id)
41423278bf5SVinod Koul {
41523278bf5SVinod Koul 	struct lt9611 *lt9611 = dev_id;
41623278bf5SVinod Koul 	unsigned int irq_flag0 = 0;
41723278bf5SVinod Koul 	unsigned int irq_flag3 = 0;
41823278bf5SVinod Koul 
41923278bf5SVinod Koul 	regmap_read(lt9611->regmap, 0x820f, &irq_flag3);
42023278bf5SVinod Koul 	regmap_read(lt9611->regmap, 0x820c, &irq_flag0);
42123278bf5SVinod Koul 
42223278bf5SVinod Koul 	/* hpd changed low */
42323278bf5SVinod Koul 	if (irq_flag3 & 0x80) {
42423278bf5SVinod Koul 		dev_info(lt9611->dev, "hdmi cable disconnected\n");
42523278bf5SVinod Koul 
42623278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8207, 0xbf);
42723278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8207, 0x3f);
42823278bf5SVinod Koul 	}
42923278bf5SVinod Koul 
43023278bf5SVinod Koul 	/* hpd changed high */
43123278bf5SVinod Koul 	if (irq_flag3 & 0x40) {
43223278bf5SVinod Koul 		dev_info(lt9611->dev, "hdmi cable connected\n");
43323278bf5SVinod Koul 
43423278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8207, 0x7f);
43523278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8207, 0x3f);
43623278bf5SVinod Koul 	}
43723278bf5SVinod Koul 
43823278bf5SVinod Koul 	if (irq_flag3 & 0xc0 && lt9611->bridge.dev)
43923278bf5SVinod Koul 		drm_kms_helper_hotplug_event(lt9611->bridge.dev);
44023278bf5SVinod Koul 
44123278bf5SVinod Koul 	/* video input changed */
44223278bf5SVinod Koul 	if (irq_flag0 & 0x01) {
44323278bf5SVinod Koul 		dev_info(lt9611->dev, "video input changed\n");
44423278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x829e, 0xff);
44523278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x829e, 0xf7);
44623278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8204, 0xff);
44723278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8204, 0xfe);
44823278bf5SVinod Koul 	}
44923278bf5SVinod Koul 
45023278bf5SVinod Koul 	return IRQ_HANDLED;
45123278bf5SVinod Koul }
45223278bf5SVinod Koul 
lt9611_enable_hpd_interrupts(struct lt9611 * lt9611)45323278bf5SVinod Koul static void lt9611_enable_hpd_interrupts(struct lt9611 *lt9611)
45423278bf5SVinod Koul {
45523278bf5SVinod Koul 	unsigned int val;
45623278bf5SVinod Koul 
45723278bf5SVinod Koul 	regmap_read(lt9611->regmap, 0x8203, &val);
45823278bf5SVinod Koul 
45923278bf5SVinod Koul 	val &= ~0xc0;
46023278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8203, val);
46123278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8207, 0xff); /* clear */
46223278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8207, 0x3f);
46323278bf5SVinod Koul }
46423278bf5SVinod Koul 
lt9611_sleep_setup(struct lt9611 * lt9611)46523278bf5SVinod Koul static void lt9611_sleep_setup(struct lt9611 *lt9611)
46623278bf5SVinod Koul {
46723278bf5SVinod Koul 	const struct reg_sequence sleep_setup[] = {
46823278bf5SVinod Koul 		{ 0x8024, 0x76 },
46923278bf5SVinod Koul 		{ 0x8023, 0x01 },
47023278bf5SVinod Koul 		{ 0x8157, 0x03 }, /* set addr pin as output */
47123278bf5SVinod Koul 		{ 0x8149, 0x0b },
472ae2d329fSDmitry Baryshkov 
47323278bf5SVinod Koul 		{ 0x8102, 0x48 }, /* MIPI Rx power down */
47423278bf5SVinod Koul 		{ 0x8123, 0x80 },
47523278bf5SVinod Koul 		{ 0x8130, 0x00 },
476ae2d329fSDmitry Baryshkov 		{ 0x8011, 0x0a },
47723278bf5SVinod Koul 	};
47823278bf5SVinod Koul 
47923278bf5SVinod Koul 	regmap_multi_reg_write(lt9611->regmap,
48023278bf5SVinod Koul 			       sleep_setup, ARRAY_SIZE(sleep_setup));
48123278bf5SVinod Koul 	lt9611->sleep = true;
48223278bf5SVinod Koul }
48323278bf5SVinod Koul 
lt9611_power_on(struct lt9611 * lt9611)48423278bf5SVinod Koul static int lt9611_power_on(struct lt9611 *lt9611)
48523278bf5SVinod Koul {
48623278bf5SVinod Koul 	int ret;
48723278bf5SVinod Koul 	const struct reg_sequence seq[] = {
48823278bf5SVinod Koul 		/* LT9611_System_Init */
48923278bf5SVinod Koul 		{ 0x8101, 0x18 }, /* sel xtal clock */
49023278bf5SVinod Koul 
49123278bf5SVinod Koul 		/* timer for frequency meter */
49223278bf5SVinod Koul 		{ 0x821b, 0x69 }, /* timer 2 */
49323278bf5SVinod Koul 		{ 0x821c, 0x78 },
49423278bf5SVinod Koul 		{ 0x82cb, 0x69 }, /* timer 1 */
49523278bf5SVinod Koul 		{ 0x82cc, 0x78 },
49623278bf5SVinod Koul 
49723278bf5SVinod Koul 		/* irq init */
49823278bf5SVinod Koul 		{ 0x8251, 0x01 },
49923278bf5SVinod Koul 		{ 0x8258, 0x0a }, /* hpd irq */
50023278bf5SVinod Koul 		{ 0x8259, 0x80 }, /* hpd debounce width */
50123278bf5SVinod Koul 		{ 0x829e, 0xf7 }, /* video check irq */
50223278bf5SVinod Koul 
50323278bf5SVinod Koul 		/* power consumption for work */
50423278bf5SVinod Koul 		{ 0x8004, 0xf0 },
50523278bf5SVinod Koul 		{ 0x8006, 0xf0 },
50623278bf5SVinod Koul 		{ 0x800a, 0x80 },
50723278bf5SVinod Koul 		{ 0x800b, 0x40 },
50823278bf5SVinod Koul 		{ 0x800d, 0xef },
50923278bf5SVinod Koul 		{ 0x8011, 0xfa },
51023278bf5SVinod Koul 	};
51123278bf5SVinod Koul 
51223278bf5SVinod Koul 	if (lt9611->power_on)
51323278bf5SVinod Koul 		return 0;
51423278bf5SVinod Koul 
51523278bf5SVinod Koul 	ret = regmap_multi_reg_write(lt9611->regmap, seq, ARRAY_SIZE(seq));
51623278bf5SVinod Koul 	if (!ret)
51723278bf5SVinod Koul 		lt9611->power_on = true;
51823278bf5SVinod Koul 
51923278bf5SVinod Koul 	return ret;
52023278bf5SVinod Koul }
52123278bf5SVinod Koul 
lt9611_power_off(struct lt9611 * lt9611)52223278bf5SVinod Koul static int lt9611_power_off(struct lt9611 *lt9611)
52323278bf5SVinod Koul {
52423278bf5SVinod Koul 	int ret;
52523278bf5SVinod Koul 
52623278bf5SVinod Koul 	ret = regmap_write(lt9611->regmap, 0x8130, 0x6a);
52723278bf5SVinod Koul 	if (!ret)
52823278bf5SVinod Koul 		lt9611->power_on = false;
52923278bf5SVinod Koul 
53023278bf5SVinod Koul 	return ret;
53123278bf5SVinod Koul }
53223278bf5SVinod Koul 
lt9611_reset(struct lt9611 * lt9611)53323278bf5SVinod Koul static void lt9611_reset(struct lt9611 *lt9611)
53423278bf5SVinod Koul {
53523278bf5SVinod Koul 	gpiod_set_value_cansleep(lt9611->reset_gpio, 1);
53623278bf5SVinod Koul 	msleep(20);
53723278bf5SVinod Koul 
53823278bf5SVinod Koul 	gpiod_set_value_cansleep(lt9611->reset_gpio, 0);
53923278bf5SVinod Koul 	msleep(20);
54023278bf5SVinod Koul 
54123278bf5SVinod Koul 	gpiod_set_value_cansleep(lt9611->reset_gpio, 1);
54223278bf5SVinod Koul 	msleep(100);
54323278bf5SVinod Koul }
54423278bf5SVinod Koul 
lt9611_assert_5v(struct lt9611 * lt9611)54523278bf5SVinod Koul static void lt9611_assert_5v(struct lt9611 *lt9611)
54623278bf5SVinod Koul {
54723278bf5SVinod Koul 	if (!lt9611->enable_gpio)
54823278bf5SVinod Koul 		return;
54923278bf5SVinod Koul 
55023278bf5SVinod Koul 	gpiod_set_value_cansleep(lt9611->enable_gpio, 1);
55123278bf5SVinod Koul 	msleep(20);
55223278bf5SVinod Koul }
55323278bf5SVinod Koul 
lt9611_regulator_init(struct lt9611 * lt9611)55423278bf5SVinod Koul static int lt9611_regulator_init(struct lt9611 *lt9611)
55523278bf5SVinod Koul {
55623278bf5SVinod Koul 	int ret;
55723278bf5SVinod Koul 
55823278bf5SVinod Koul 	lt9611->supplies[0].supply = "vdd";
55923278bf5SVinod Koul 	lt9611->supplies[1].supply = "vcc";
56023278bf5SVinod Koul 
56123278bf5SVinod Koul 	ret = devm_regulator_bulk_get(lt9611->dev, 2, lt9611->supplies);
56223278bf5SVinod Koul 	if (ret < 0)
56323278bf5SVinod Koul 		return ret;
56423278bf5SVinod Koul 
56523278bf5SVinod Koul 	return regulator_set_load(lt9611->supplies[0].consumer, 300000);
56623278bf5SVinod Koul }
56723278bf5SVinod Koul 
lt9611_regulator_enable(struct lt9611 * lt9611)56823278bf5SVinod Koul static int lt9611_regulator_enable(struct lt9611 *lt9611)
56923278bf5SVinod Koul {
57023278bf5SVinod Koul 	int ret;
57123278bf5SVinod Koul 
57223278bf5SVinod Koul 	ret = regulator_enable(lt9611->supplies[0].consumer);
57323278bf5SVinod Koul 	if (ret < 0)
57423278bf5SVinod Koul 		return ret;
57523278bf5SVinod Koul 
57623278bf5SVinod Koul 	usleep_range(1000, 10000);
57723278bf5SVinod Koul 
57823278bf5SVinod Koul 	ret = regulator_enable(lt9611->supplies[1].consumer);
57923278bf5SVinod Koul 	if (ret < 0) {
58023278bf5SVinod Koul 		regulator_disable(lt9611->supplies[0].consumer);
58123278bf5SVinod Koul 		return ret;
58223278bf5SVinod Koul 	}
58323278bf5SVinod Koul 
58423278bf5SVinod Koul 	return 0;
58523278bf5SVinod Koul }
58623278bf5SVinod Koul 
lt9611_bridge_detect(struct drm_bridge * bridge)5870c3997b0SDmitry Baryshkov static enum drm_connector_status lt9611_bridge_detect(struct drm_bridge *bridge)
58823278bf5SVinod Koul {
5890c3997b0SDmitry Baryshkov 	struct lt9611 *lt9611 = bridge_to_lt9611(bridge);
59023278bf5SVinod Koul 	unsigned int reg_val = 0;
59123278bf5SVinod Koul 	int connected = 0;
59223278bf5SVinod Koul 
59323278bf5SVinod Koul 	regmap_read(lt9611->regmap, 0x825e, &reg_val);
594649eb382SJohn Stultz 	connected  = (reg_val & (BIT(2) | BIT(0)));
59523278bf5SVinod Koul 
59623278bf5SVinod Koul 	lt9611->status = connected ?  connector_status_connected :
59723278bf5SVinod Koul 				connector_status_disconnected;
59823278bf5SVinod Koul 
59923278bf5SVinod Koul 	return lt9611->status;
60023278bf5SVinod Koul }
60123278bf5SVinod Koul 
lt9611_read_edid(struct lt9611 * lt9611)60223278bf5SVinod Koul static int lt9611_read_edid(struct lt9611 *lt9611)
60323278bf5SVinod Koul {
60423278bf5SVinod Koul 	unsigned int temp;
60523278bf5SVinod Koul 	int ret = 0;
60623278bf5SVinod Koul 	int i, j;
60723278bf5SVinod Koul 
60823278bf5SVinod Koul 	/* memset to clear old buffer, if any */
60923278bf5SVinod Koul 	memset(lt9611->edid_buf, 0, sizeof(lt9611->edid_buf));
61023278bf5SVinod Koul 
61123278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8503, 0xc9);
61223278bf5SVinod Koul 
61323278bf5SVinod Koul 	/* 0xA0 is EDID device address */
61423278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8504, 0xa0);
61523278bf5SVinod Koul 	/* 0x00 is EDID offset address */
61623278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8505, 0x00);
61723278bf5SVinod Koul 
61823278bf5SVinod Koul 	/* length for read */
61923278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8506, EDID_LEN);
62023278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8514, 0x7f);
62123278bf5SVinod Koul 
62223278bf5SVinod Koul 	for (i = 0; i < EDID_LOOP; i++) {
62323278bf5SVinod Koul 		/* offset address */
62423278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8505, i * EDID_LEN);
62523278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8507, 0x36);
62623278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8507, 0x31);
62723278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x8507, 0x37);
62823278bf5SVinod Koul 		usleep_range(5000, 10000);
62923278bf5SVinod Koul 
63023278bf5SVinod Koul 		regmap_read(lt9611->regmap, 0x8540, &temp);
63123278bf5SVinod Koul 
63223278bf5SVinod Koul 		if (temp & KEY_DDC_ACCS_DONE) {
63323278bf5SVinod Koul 			for (j = 0; j < EDID_LEN; j++) {
63423278bf5SVinod Koul 				regmap_read(lt9611->regmap, 0x8583, &temp);
63523278bf5SVinod Koul 				lt9611->edid_buf[i * EDID_LEN + j] = temp;
63623278bf5SVinod Koul 			}
63723278bf5SVinod Koul 
63823278bf5SVinod Koul 		} else if (temp & DDC_NO_ACK) { /* DDC No Ack or Abitration lost */
63923278bf5SVinod Koul 			dev_err(lt9611->dev, "read edid failed: no ack\n");
64023278bf5SVinod Koul 			ret = -EIO;
64123278bf5SVinod Koul 			goto end;
64223278bf5SVinod Koul 
64323278bf5SVinod Koul 		} else {
64423278bf5SVinod Koul 			dev_err(lt9611->dev, "read edid failed: access not done\n");
64523278bf5SVinod Koul 			ret = -EIO;
64623278bf5SVinod Koul 			goto end;
64723278bf5SVinod Koul 		}
64823278bf5SVinod Koul 	}
64923278bf5SVinod Koul 
65023278bf5SVinod Koul end:
65123278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8507, 0x1f);
65223278bf5SVinod Koul 	return ret;
65323278bf5SVinod Koul }
65423278bf5SVinod Koul 
65523278bf5SVinod Koul static int
lt9611_get_edid_block(void * data,u8 * buf,unsigned int block,size_t len)65623278bf5SVinod Koul lt9611_get_edid_block(void *data, u8 *buf, unsigned int block, size_t len)
65723278bf5SVinod Koul {
65823278bf5SVinod Koul 	struct lt9611 *lt9611 = data;
65923278bf5SVinod Koul 	int ret;
66023278bf5SVinod Koul 
66123278bf5SVinod Koul 	if (len > 128)
66223278bf5SVinod Koul 		return -EINVAL;
66323278bf5SVinod Koul 
66423278bf5SVinod Koul 	/* supports up to 1 extension block */
66523278bf5SVinod Koul 	/* TODO: add support for more extension blocks */
66623278bf5SVinod Koul 	if (block > 1)
66723278bf5SVinod Koul 		return -EINVAL;
66823278bf5SVinod Koul 
66923278bf5SVinod Koul 	if (block == 0) {
67023278bf5SVinod Koul 		ret = lt9611_read_edid(lt9611);
67123278bf5SVinod Koul 		if (ret) {
67223278bf5SVinod Koul 			dev_err(lt9611->dev, "edid read failed\n");
67323278bf5SVinod Koul 			return ret;
67423278bf5SVinod Koul 		}
67523278bf5SVinod Koul 	}
67623278bf5SVinod Koul 
67723278bf5SVinod Koul 	block %= 2;
67823278bf5SVinod Koul 	memcpy(buf, lt9611->edid_buf + (block * 128), len);
67923278bf5SVinod Koul 
68023278bf5SVinod Koul 	return 0;
68123278bf5SVinod Koul }
68223278bf5SVinod Koul 
68323278bf5SVinod Koul /* bridge funcs */
68497d7ca79SMarek Vasut static void
lt9611_bridge_atomic_enable(struct drm_bridge * bridge,struct drm_bridge_state * old_bridge_state)68597d7ca79SMarek Vasut lt9611_bridge_atomic_enable(struct drm_bridge *bridge,
68697d7ca79SMarek Vasut 			    struct drm_bridge_state *old_bridge_state)
68723278bf5SVinod Koul {
68823278bf5SVinod Koul 	struct lt9611 *lt9611 = bridge_to_lt9611(bridge);
689fad97f28SDmitry Baryshkov 	struct drm_atomic_state *state = old_bridge_state->base.state;
690fad97f28SDmitry Baryshkov 	struct drm_connector *connector;
691fad97f28SDmitry Baryshkov 	struct drm_connector_state *conn_state;
692fad97f28SDmitry Baryshkov 	struct drm_crtc_state *crtc_state;
693fad97f28SDmitry Baryshkov 	struct drm_display_mode *mode;
694fad97f28SDmitry Baryshkov 	unsigned int postdiv;
695fad97f28SDmitry Baryshkov 
696fad97f28SDmitry Baryshkov 	connector = drm_atomic_get_new_connector_for_encoder(state, bridge->encoder);
697fad97f28SDmitry Baryshkov 	if (WARN_ON(!connector))
698fad97f28SDmitry Baryshkov 		return;
699fad97f28SDmitry Baryshkov 
700fad97f28SDmitry Baryshkov 	conn_state = drm_atomic_get_new_connector_state(state, connector);
701fad97f28SDmitry Baryshkov 	if (WARN_ON(!conn_state))
702fad97f28SDmitry Baryshkov 		return;
703fad97f28SDmitry Baryshkov 
704fad97f28SDmitry Baryshkov 	crtc_state = drm_atomic_get_new_crtc_state(state, conn_state->crtc);
705fad97f28SDmitry Baryshkov 	if (WARN_ON(!crtc_state))
706fad97f28SDmitry Baryshkov 		return;
707fad97f28SDmitry Baryshkov 
708fad97f28SDmitry Baryshkov 	mode = &crtc_state->adjusted_mode;
709fad97f28SDmitry Baryshkov 
710fad97f28SDmitry Baryshkov 	lt9611_mipi_input_digital(lt9611, mode);
711fad97f28SDmitry Baryshkov 	lt9611_pll_setup(lt9611, mode, &postdiv);
712fad97f28SDmitry Baryshkov 	lt9611_mipi_video_setup(lt9611, mode);
713fad97f28SDmitry Baryshkov 	lt9611_pcr_setup(lt9611, mode, postdiv);
714fad97f28SDmitry Baryshkov 
71523278bf5SVinod Koul 	if (lt9611_power_on(lt9611)) {
71623278bf5SVinod Koul 		dev_err(lt9611->dev, "power on failed\n");
71723278bf5SVinod Koul 		return;
71823278bf5SVinod Koul 	}
71923278bf5SVinod Koul 
72023278bf5SVinod Koul 	lt9611_mipi_input_analog(lt9611);
72184cf74d9SDmitry Baryshkov 	lt9611_hdmi_set_infoframes(lt9611, connector, mode);
7220c747469SDmitry Baryshkov 	lt9611_hdmi_tx_digital(lt9611, connector->display_info.is_hdmi);
72323278bf5SVinod Koul 	lt9611_hdmi_tx_phy(lt9611);
72423278bf5SVinod Koul 
72523278bf5SVinod Koul 	msleep(500);
72623278bf5SVinod Koul 
72723278bf5SVinod Koul 	lt9611_video_check(lt9611);
72823278bf5SVinod Koul 
72923278bf5SVinod Koul 	/* Enable HDMI output */
73023278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8130, 0xea);
73123278bf5SVinod Koul }
73223278bf5SVinod Koul 
73397d7ca79SMarek Vasut static void
lt9611_bridge_atomic_disable(struct drm_bridge * bridge,struct drm_bridge_state * old_bridge_state)73497d7ca79SMarek Vasut lt9611_bridge_atomic_disable(struct drm_bridge *bridge,
73597d7ca79SMarek Vasut 			     struct drm_bridge_state *old_bridge_state)
73623278bf5SVinod Koul {
73723278bf5SVinod Koul 	struct lt9611 *lt9611 = bridge_to_lt9611(bridge);
73823278bf5SVinod Koul 	int ret;
73923278bf5SVinod Koul 
74023278bf5SVinod Koul 	/* Disable HDMI output */
74123278bf5SVinod Koul 	ret = regmap_write(lt9611->regmap, 0x8130, 0x6a);
74223278bf5SVinod Koul 	if (ret) {
74323278bf5SVinod Koul 		dev_err(lt9611->dev, "video on failed\n");
74423278bf5SVinod Koul 		return;
74523278bf5SVinod Koul 	}
74623278bf5SVinod Koul 
74723278bf5SVinod Koul 	if (lt9611_power_off(lt9611)) {
74823278bf5SVinod Koul 		dev_err(lt9611->dev, "power on failed\n");
74923278bf5SVinod Koul 		return;
75023278bf5SVinod Koul 	}
75123278bf5SVinod Koul }
75223278bf5SVinod Koul 
lt9611_attach_dsi(struct lt9611 * lt9611,struct device_node * dsi_node)75323278bf5SVinod Koul static struct mipi_dsi_device *lt9611_attach_dsi(struct lt9611 *lt9611,
75423278bf5SVinod Koul 						 struct device_node *dsi_node)
75523278bf5SVinod Koul {
756b0a7f873SDmitry Baryshkov 	const struct mipi_dsi_device_info info = { "lt9611", 0, lt9611->dev->of_node};
75723278bf5SVinod Koul 	struct mipi_dsi_device *dsi;
75823278bf5SVinod Koul 	struct mipi_dsi_host *host;
759b91df118SMaxime Ripard 	struct device *dev = lt9611->dev;
76023278bf5SVinod Koul 	int ret;
76123278bf5SVinod Koul 
76223278bf5SVinod Koul 	host = of_find_mipi_dsi_host_by_node(dsi_node);
763*ae2ac0aeSNícolas F. R. A. Prado 	if (!host)
764*ae2ac0aeSNícolas F. R. A. Prado 		return ERR_PTR(dev_err_probe(lt9611->dev, -EPROBE_DEFER, "failed to find dsi host\n"));
76523278bf5SVinod Koul 
766b91df118SMaxime Ripard 	dsi = devm_mipi_dsi_device_register_full(dev, host, &info);
76723278bf5SVinod Koul 	if (IS_ERR(dsi)) {
76823278bf5SVinod Koul 		dev_err(lt9611->dev, "failed to create dsi device\n");
76923278bf5SVinod Koul 		return dsi;
77023278bf5SVinod Koul 	}
77123278bf5SVinod Koul 
77223278bf5SVinod Koul 	dsi->lanes = 4;
77323278bf5SVinod Koul 	dsi->format = MIPI_DSI_FMT_RGB888;
77423278bf5SVinod Koul 	dsi->mode_flags = MIPI_DSI_MODE_VIDEO | MIPI_DSI_MODE_VIDEO_SYNC_PULSE |
7753c6bd1b7SNeil Armstrong 			  MIPI_DSI_MODE_VIDEO_HSE;
77623278bf5SVinod Koul 
777b91df118SMaxime Ripard 	ret = devm_mipi_dsi_attach(dev, dsi);
77823278bf5SVinod Koul 	if (ret < 0) {
779b91df118SMaxime Ripard 		dev_err(dev, "failed to attach dsi to host\n");
78023278bf5SVinod Koul 		return ERR_PTR(ret);
78123278bf5SVinod Koul 	}
78223278bf5SVinod Koul 
78323278bf5SVinod Koul 	return dsi;
78423278bf5SVinod Koul }
78523278bf5SVinod Koul 
lt9611_bridge_attach(struct drm_bridge * bridge,enum drm_bridge_attach_flags flags)78623278bf5SVinod Koul static int lt9611_bridge_attach(struct drm_bridge *bridge,
78723278bf5SVinod Koul 				enum drm_bridge_attach_flags flags)
78823278bf5SVinod Koul {
78923278bf5SVinod Koul 	struct lt9611 *lt9611 = bridge_to_lt9611(bridge);
79023278bf5SVinod Koul 
7910c3997b0SDmitry Baryshkov 	return drm_bridge_attach(bridge->encoder, lt9611->next_bridge,
7920c3997b0SDmitry Baryshkov 				 bridge, flags);
79323278bf5SVinod Koul }
79423278bf5SVinod Koul 
lt9611_bridge_mode_valid(struct drm_bridge * bridge,const struct drm_display_info * info,const struct drm_display_mode * mode)79523278bf5SVinod Koul static enum drm_mode_status lt9611_bridge_mode_valid(struct drm_bridge *bridge,
79623278bf5SVinod Koul 						     const struct drm_display_info *info,
79723278bf5SVinod Koul 						     const struct drm_display_mode *mode)
79823278bf5SVinod Koul {
799d1a97648SRobert Foss 	struct lt9611 *lt9611 = bridge_to_lt9611(bridge);
80023278bf5SVinod Koul 
8014914cbc4SDmitry Baryshkov 	if (mode->hdisplay > 3840)
8024914cbc4SDmitry Baryshkov 		return MODE_BAD_HVALUE;
8034914cbc4SDmitry Baryshkov 
8044914cbc4SDmitry Baryshkov 	if (mode->vdisplay > 2160)
8054914cbc4SDmitry Baryshkov 		return MODE_BAD_VVALUE;
8064914cbc4SDmitry Baryshkov 
8074914cbc4SDmitry Baryshkov 	if (mode->hdisplay == 3840 &&
8084914cbc4SDmitry Baryshkov 	    mode->vdisplay == 2160 &&
8094914cbc4SDmitry Baryshkov 	    drm_mode_vrefresh(mode) > 30)
8104914cbc4SDmitry Baryshkov 		return MODE_CLOCK_HIGH;
8114914cbc4SDmitry Baryshkov 
8124914cbc4SDmitry Baryshkov 	if (mode->hdisplay > 2000 && !lt9611->dsi1_node)
813d1a97648SRobert Foss 		return MODE_PANEL;
814d1a97648SRobert Foss 	else
815d1a97648SRobert Foss 		return MODE_OK;
81623278bf5SVinod Koul }
81723278bf5SVinod Koul 
lt9611_bridge_atomic_pre_enable(struct drm_bridge * bridge,struct drm_bridge_state * old_bridge_state)818fad97f28SDmitry Baryshkov static void lt9611_bridge_atomic_pre_enable(struct drm_bridge *bridge,
819fad97f28SDmitry Baryshkov 					    struct drm_bridge_state *old_bridge_state)
82023278bf5SVinod Koul {
82123278bf5SVinod Koul 	struct lt9611 *lt9611 = bridge_to_lt9611(bridge);
822a7790f6bSDmitry Baryshkov 	static const struct reg_sequence reg_cfg[] = {
823a7790f6bSDmitry Baryshkov 		{ 0x8102, 0x12 },
824a7790f6bSDmitry Baryshkov 		{ 0x8123, 0x40 },
825a7790f6bSDmitry Baryshkov 		{ 0x8130, 0xea },
826a7790f6bSDmitry Baryshkov 		{ 0x8011, 0xfa },
827a7790f6bSDmitry Baryshkov 	};
82823278bf5SVinod Koul 
82923278bf5SVinod Koul 	if (!lt9611->sleep)
83023278bf5SVinod Koul 		return;
83123278bf5SVinod Koul 
832a7790f6bSDmitry Baryshkov 	regmap_multi_reg_write(lt9611->regmap,
833a7790f6bSDmitry Baryshkov 			       reg_cfg, ARRAY_SIZE(reg_cfg));
83423278bf5SVinod Koul 
83523278bf5SVinod Koul 	lt9611->sleep = false;
83623278bf5SVinod Koul }
83723278bf5SVinod Koul 
83897d7ca79SMarek Vasut static void
lt9611_bridge_atomic_post_disable(struct drm_bridge * bridge,struct drm_bridge_state * old_bridge_state)83997d7ca79SMarek Vasut lt9611_bridge_atomic_post_disable(struct drm_bridge *bridge,
84097d7ca79SMarek Vasut 				  struct drm_bridge_state *old_bridge_state)
84123278bf5SVinod Koul {
84223278bf5SVinod Koul 	struct lt9611 *lt9611 = bridge_to_lt9611(bridge);
84323278bf5SVinod Koul 
84423278bf5SVinod Koul 	lt9611_sleep_setup(lt9611);
84523278bf5SVinod Koul }
84623278bf5SVinod Koul 
lt9611_bridge_get_edid(struct drm_bridge * bridge,struct drm_connector * connector)84723278bf5SVinod Koul static struct edid *lt9611_bridge_get_edid(struct drm_bridge *bridge,
84823278bf5SVinod Koul 					   struct drm_connector *connector)
84923278bf5SVinod Koul {
85023278bf5SVinod Koul 	struct lt9611 *lt9611 = bridge_to_lt9611(bridge);
85123278bf5SVinod Koul 
85223278bf5SVinod Koul 	lt9611_power_on(lt9611);
85323278bf5SVinod Koul 	return drm_do_get_edid(connector, lt9611_get_edid_block, lt9611);
85423278bf5SVinod Koul }
85523278bf5SVinod Koul 
lt9611_bridge_hpd_enable(struct drm_bridge * bridge)85623278bf5SVinod Koul static void lt9611_bridge_hpd_enable(struct drm_bridge *bridge)
85723278bf5SVinod Koul {
85823278bf5SVinod Koul 	struct lt9611 *lt9611 = bridge_to_lt9611(bridge);
85923278bf5SVinod Koul 
86023278bf5SVinod Koul 	lt9611_enable_hpd_interrupts(lt9611);
86123278bf5SVinod Koul }
86223278bf5SVinod Koul 
863ee2f7c9dSMarek Vasut #define MAX_INPUT_SEL_FORMATS	1
864ee2f7c9dSMarek Vasut 
865ee2f7c9dSMarek Vasut static u32 *
lt9611_atomic_get_input_bus_fmts(struct drm_bridge * bridge,struct drm_bridge_state * bridge_state,struct drm_crtc_state * crtc_state,struct drm_connector_state * conn_state,u32 output_fmt,unsigned int * num_input_fmts)866ee2f7c9dSMarek Vasut lt9611_atomic_get_input_bus_fmts(struct drm_bridge *bridge,
867ee2f7c9dSMarek Vasut 				 struct drm_bridge_state *bridge_state,
868ee2f7c9dSMarek Vasut 				 struct drm_crtc_state *crtc_state,
869ee2f7c9dSMarek Vasut 				 struct drm_connector_state *conn_state,
870ee2f7c9dSMarek Vasut 				 u32 output_fmt,
871ee2f7c9dSMarek Vasut 				 unsigned int *num_input_fmts)
872ee2f7c9dSMarek Vasut {
873ee2f7c9dSMarek Vasut 	u32 *input_fmts;
874ee2f7c9dSMarek Vasut 
875ee2f7c9dSMarek Vasut 	*num_input_fmts = 0;
876ee2f7c9dSMarek Vasut 
877ee2f7c9dSMarek Vasut 	input_fmts = kcalloc(MAX_INPUT_SEL_FORMATS, sizeof(*input_fmts),
878ee2f7c9dSMarek Vasut 			     GFP_KERNEL);
879ee2f7c9dSMarek Vasut 	if (!input_fmts)
880ee2f7c9dSMarek Vasut 		return NULL;
881ee2f7c9dSMarek Vasut 
882ee2f7c9dSMarek Vasut 	/* This is the DSI-end bus format */
883ee2f7c9dSMarek Vasut 	input_fmts[0] = MEDIA_BUS_FMT_RGB888_1X24;
884ee2f7c9dSMarek Vasut 	*num_input_fmts = 1;
885ee2f7c9dSMarek Vasut 
886ee2f7c9dSMarek Vasut 	return input_fmts;
887ee2f7c9dSMarek Vasut }
888ee2f7c9dSMarek Vasut 
88923278bf5SVinod Koul static const struct drm_bridge_funcs lt9611_bridge_funcs = {
89023278bf5SVinod Koul 	.attach = lt9611_bridge_attach,
89123278bf5SVinod Koul 	.mode_valid = lt9611_bridge_mode_valid,
89223278bf5SVinod Koul 	.detect = lt9611_bridge_detect,
89323278bf5SVinod Koul 	.get_edid = lt9611_bridge_get_edid,
89423278bf5SVinod Koul 	.hpd_enable = lt9611_bridge_hpd_enable,
89597d7ca79SMarek Vasut 
896fad97f28SDmitry Baryshkov 	.atomic_pre_enable = lt9611_bridge_atomic_pre_enable,
89797d7ca79SMarek Vasut 	.atomic_enable = lt9611_bridge_atomic_enable,
89897d7ca79SMarek Vasut 	.atomic_disable = lt9611_bridge_atomic_disable,
89997d7ca79SMarek Vasut 	.atomic_post_disable = lt9611_bridge_atomic_post_disable,
90097d7ca79SMarek Vasut 	.atomic_duplicate_state = drm_atomic_helper_bridge_duplicate_state,
90197d7ca79SMarek Vasut 	.atomic_destroy_state = drm_atomic_helper_bridge_destroy_state,
90297d7ca79SMarek Vasut 	.atomic_reset = drm_atomic_helper_bridge_reset,
903ee2f7c9dSMarek Vasut 	.atomic_get_input_bus_fmts = lt9611_atomic_get_input_bus_fmts,
90423278bf5SVinod Koul };
90523278bf5SVinod Koul 
lt9611_parse_dt(struct device * dev,struct lt9611 * lt9611)90623278bf5SVinod Koul static int lt9611_parse_dt(struct device *dev,
90723278bf5SVinod Koul 			   struct lt9611 *lt9611)
90823278bf5SVinod Koul {
90985a51673SVinod Koul 	lt9611->dsi0_node = of_graph_get_remote_node(dev->of_node, 0, -1);
91023278bf5SVinod Koul 	if (!lt9611->dsi0_node) {
91123278bf5SVinod Koul 		dev_err(lt9611->dev, "failed to get remote node for primary dsi\n");
91223278bf5SVinod Koul 		return -ENODEV;
91323278bf5SVinod Koul 	}
91423278bf5SVinod Koul 
91585a51673SVinod Koul 	lt9611->dsi1_node = of_graph_get_remote_node(dev->of_node, 1, -1);
91623278bf5SVinod Koul 
91723278bf5SVinod Koul 	lt9611->ac_mode = of_property_read_bool(dev->of_node, "lt,ac-mode");
91823278bf5SVinod Koul 
9190c3997b0SDmitry Baryshkov 	return drm_of_find_panel_or_bridge(dev->of_node, 2, -1, NULL, &lt9611->next_bridge);
92023278bf5SVinod Koul }
92123278bf5SVinod Koul 
lt9611_gpio_init(struct lt9611 * lt9611)92223278bf5SVinod Koul static int lt9611_gpio_init(struct lt9611 *lt9611)
92323278bf5SVinod Koul {
92423278bf5SVinod Koul 	struct device *dev = lt9611->dev;
92523278bf5SVinod Koul 
92623278bf5SVinod Koul 	lt9611->reset_gpio = devm_gpiod_get(dev, "reset", GPIOD_OUT_HIGH);
92723278bf5SVinod Koul 	if (IS_ERR(lt9611->reset_gpio)) {
92823278bf5SVinod Koul 		dev_err(dev, "failed to acquire reset gpio\n");
92923278bf5SVinod Koul 		return PTR_ERR(lt9611->reset_gpio);
93023278bf5SVinod Koul 	}
93123278bf5SVinod Koul 
93223278bf5SVinod Koul 	lt9611->enable_gpio = devm_gpiod_get_optional(dev, "enable",
93323278bf5SVinod Koul 						      GPIOD_OUT_LOW);
93423278bf5SVinod Koul 	if (IS_ERR(lt9611->enable_gpio)) {
93523278bf5SVinod Koul 		dev_err(dev, "failed to acquire enable gpio\n");
93623278bf5SVinod Koul 		return PTR_ERR(lt9611->enable_gpio);
93723278bf5SVinod Koul 	}
93823278bf5SVinod Koul 
93923278bf5SVinod Koul 	return 0;
94023278bf5SVinod Koul }
94123278bf5SVinod Koul 
lt9611_read_device_rev(struct lt9611 * lt9611)94223278bf5SVinod Koul static int lt9611_read_device_rev(struct lt9611 *lt9611)
94323278bf5SVinod Koul {
94423278bf5SVinod Koul 	unsigned int rev;
94523278bf5SVinod Koul 	int ret;
94623278bf5SVinod Koul 
94723278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x80ee, 0x01);
94823278bf5SVinod Koul 	ret = regmap_read(lt9611->regmap, 0x8002, &rev);
94923278bf5SVinod Koul 	if (ret)
95023278bf5SVinod Koul 		dev_err(lt9611->dev, "failed to read revision: %d\n", ret);
95123278bf5SVinod Koul 	else
95223278bf5SVinod Koul 		dev_info(lt9611->dev, "LT9611 revision: 0x%x\n", rev);
95323278bf5SVinod Koul 
95423278bf5SVinod Koul 	return ret;
95523278bf5SVinod Koul }
95623278bf5SVinod Koul 
lt9611_hdmi_hw_params(struct device * dev,void * data,struct hdmi_codec_daifmt * fmt,struct hdmi_codec_params * hparms)95723278bf5SVinod Koul static int lt9611_hdmi_hw_params(struct device *dev, void *data,
95823278bf5SVinod Koul 				 struct hdmi_codec_daifmt *fmt,
95923278bf5SVinod Koul 				 struct hdmi_codec_params *hparms)
96023278bf5SVinod Koul {
96123278bf5SVinod Koul 	struct lt9611 *lt9611 = data;
96223278bf5SVinod Koul 
96323278bf5SVinod Koul 	if (hparms->sample_rate == 48000)
96423278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x840f, 0x2b);
96523278bf5SVinod Koul 	else if (hparms->sample_rate == 96000)
96623278bf5SVinod Koul 		regmap_write(lt9611->regmap, 0x840f, 0xab);
96723278bf5SVinod Koul 	else
96823278bf5SVinod Koul 		return -EINVAL;
96923278bf5SVinod Koul 
97023278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8435, 0x00);
97123278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8436, 0x18);
97223278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8437, 0x00);
97323278bf5SVinod Koul 
97423278bf5SVinod Koul 	return 0;
97523278bf5SVinod Koul }
97623278bf5SVinod Koul 
lt9611_audio_startup(struct device * dev,void * data)97723278bf5SVinod Koul static int lt9611_audio_startup(struct device *dev, void *data)
97823278bf5SVinod Koul {
97923278bf5SVinod Koul 	struct lt9611 *lt9611 = data;
98023278bf5SVinod Koul 
98123278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x82d6, 0x8c);
98223278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x82d7, 0x04);
98323278bf5SVinod Koul 
98423278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8406, 0x08);
98523278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8407, 0x10);
98623278bf5SVinod Koul 
98723278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8434, 0xd5);
98823278bf5SVinod Koul 
98923278bf5SVinod Koul 	return 0;
99023278bf5SVinod Koul }
99123278bf5SVinod Koul 
lt9611_audio_shutdown(struct device * dev,void * data)99223278bf5SVinod Koul static void lt9611_audio_shutdown(struct device *dev, void *data)
99323278bf5SVinod Koul {
99423278bf5SVinod Koul 	struct lt9611 *lt9611 = data;
99523278bf5SVinod Koul 
99623278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8406, 0x00);
99723278bf5SVinod Koul 	regmap_write(lt9611->regmap, 0x8407, 0x00);
99823278bf5SVinod Koul }
99923278bf5SVinod Koul 
lt9611_hdmi_i2s_get_dai_id(struct snd_soc_component * component,struct device_node * endpoint)100023278bf5SVinod Koul static int lt9611_hdmi_i2s_get_dai_id(struct snd_soc_component *component,
100123278bf5SVinod Koul 				      struct device_node *endpoint)
100223278bf5SVinod Koul {
100323278bf5SVinod Koul 	struct of_endpoint of_ep;
100423278bf5SVinod Koul 	int ret;
100523278bf5SVinod Koul 
100623278bf5SVinod Koul 	ret = of_graph_parse_endpoint(endpoint, &of_ep);
100723278bf5SVinod Koul 	if (ret < 0)
100823278bf5SVinod Koul 		return ret;
100923278bf5SVinod Koul 
101023278bf5SVinod Koul 	/*
101123278bf5SVinod Koul 	 * HDMI sound should be located as reg = <2>
101223278bf5SVinod Koul 	 * Then, it is sound port 0
101323278bf5SVinod Koul 	 */
101423278bf5SVinod Koul 	if (of_ep.port == 2)
101523278bf5SVinod Koul 		return 0;
101623278bf5SVinod Koul 
101723278bf5SVinod Koul 	return -EINVAL;
101823278bf5SVinod Koul }
101923278bf5SVinod Koul 
102023278bf5SVinod Koul static const struct hdmi_codec_ops lt9611_codec_ops = {
102123278bf5SVinod Koul 	.hw_params	= lt9611_hdmi_hw_params,
102223278bf5SVinod Koul 	.audio_shutdown = lt9611_audio_shutdown,
102323278bf5SVinod Koul 	.audio_startup	= lt9611_audio_startup,
102423278bf5SVinod Koul 	.get_dai_id	= lt9611_hdmi_i2s_get_dai_id,
102523278bf5SVinod Koul };
102623278bf5SVinod Koul 
102723278bf5SVinod Koul static struct hdmi_codec_pdata codec_data = {
102823278bf5SVinod Koul 	.ops = &lt9611_codec_ops,
102923278bf5SVinod Koul 	.max_i2s_channels = 8,
103023278bf5SVinod Koul 	.i2s = 1,
103123278bf5SVinod Koul };
103223278bf5SVinod Koul 
lt9611_audio_init(struct device * dev,struct lt9611 * lt9611)103323278bf5SVinod Koul static int lt9611_audio_init(struct device *dev, struct lt9611 *lt9611)
103423278bf5SVinod Koul {
103523278bf5SVinod Koul 	codec_data.data = lt9611;
103623278bf5SVinod Koul 	lt9611->audio_pdev =
103723278bf5SVinod Koul 		platform_device_register_data(dev, HDMI_CODEC_DRV_NAME,
103823278bf5SVinod Koul 					      PLATFORM_DEVID_AUTO,
103923278bf5SVinod Koul 					      &codec_data, sizeof(codec_data));
104023278bf5SVinod Koul 
104123278bf5SVinod Koul 	return PTR_ERR_OR_ZERO(lt9611->audio_pdev);
104223278bf5SVinod Koul }
104323278bf5SVinod Koul 
lt9611_audio_exit(struct lt9611 * lt9611)104423278bf5SVinod Koul static void lt9611_audio_exit(struct lt9611 *lt9611)
104523278bf5SVinod Koul {
104623278bf5SVinod Koul 	if (lt9611->audio_pdev) {
104723278bf5SVinod Koul 		platform_device_unregister(lt9611->audio_pdev);
104823278bf5SVinod Koul 		lt9611->audio_pdev = NULL;
104923278bf5SVinod Koul 	}
105023278bf5SVinod Koul }
105123278bf5SVinod Koul 
lt9611_probe(struct i2c_client * client)1052453d0603SUwe Kleine-König static int lt9611_probe(struct i2c_client *client)
105323278bf5SVinod Koul {
105423278bf5SVinod Koul 	struct lt9611 *lt9611;
105523278bf5SVinod Koul 	struct device *dev = &client->dev;
105623278bf5SVinod Koul 	int ret;
105723278bf5SVinod Koul 
105823278bf5SVinod Koul 	if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) {
105923278bf5SVinod Koul 		dev_err(dev, "device doesn't support I2C\n");
106023278bf5SVinod Koul 		return -ENODEV;
106123278bf5SVinod Koul 	}
106223278bf5SVinod Koul 
106323278bf5SVinod Koul 	lt9611 = devm_kzalloc(dev, sizeof(*lt9611), GFP_KERNEL);
106423278bf5SVinod Koul 	if (!lt9611)
106523278bf5SVinod Koul 		return -ENOMEM;
106623278bf5SVinod Koul 
1067cc37b88bSZhiming Liu 	lt9611->dev = dev;
106823278bf5SVinod Koul 	lt9611->client = client;
106923278bf5SVinod Koul 	lt9611->sleep = false;
107023278bf5SVinod Koul 
107123278bf5SVinod Koul 	lt9611->regmap = devm_regmap_init_i2c(client, &lt9611_regmap_config);
107223278bf5SVinod Koul 	if (IS_ERR(lt9611->regmap)) {
107323278bf5SVinod Koul 		dev_err(lt9611->dev, "regmap i2c init failed\n");
107423278bf5SVinod Koul 		return PTR_ERR(lt9611->regmap);
107523278bf5SVinod Koul 	}
107623278bf5SVinod Koul 
1077cc37b88bSZhiming Liu 	ret = lt9611_parse_dt(dev, lt9611);
107823278bf5SVinod Koul 	if (ret) {
107923278bf5SVinod Koul 		dev_err(dev, "failed to parse device tree\n");
108023278bf5SVinod Koul 		return ret;
108123278bf5SVinod Koul 	}
108223278bf5SVinod Koul 
108323278bf5SVinod Koul 	ret = lt9611_gpio_init(lt9611);
108423278bf5SVinod Koul 	if (ret < 0)
108523278bf5SVinod Koul 		goto err_of_put;
108623278bf5SVinod Koul 
108723278bf5SVinod Koul 	ret = lt9611_regulator_init(lt9611);
108823278bf5SVinod Koul 	if (ret < 0)
108923278bf5SVinod Koul 		goto err_of_put;
109023278bf5SVinod Koul 
109123278bf5SVinod Koul 	lt9611_assert_5v(lt9611);
109223278bf5SVinod Koul 
109323278bf5SVinod Koul 	ret = lt9611_regulator_enable(lt9611);
109423278bf5SVinod Koul 	if (ret)
109523278bf5SVinod Koul 		goto err_of_put;
109623278bf5SVinod Koul 
109723278bf5SVinod Koul 	lt9611_reset(lt9611);
109823278bf5SVinod Koul 
109923278bf5SVinod Koul 	ret = lt9611_read_device_rev(lt9611);
110023278bf5SVinod Koul 	if (ret) {
110123278bf5SVinod Koul 		dev_err(dev, "failed to read chip rev\n");
110223278bf5SVinod Koul 		goto err_disable_regulators;
110323278bf5SVinod Koul 	}
110423278bf5SVinod Koul 
110523278bf5SVinod Koul 	ret = devm_request_threaded_irq(dev, client->irq, NULL,
110623278bf5SVinod Koul 					lt9611_irq_thread_handler,
110723278bf5SVinod Koul 					IRQF_ONESHOT, "lt9611", lt9611);
110823278bf5SVinod Koul 	if (ret) {
110923278bf5SVinod Koul 		dev_err(dev, "failed to request irq\n");
111023278bf5SVinod Koul 		goto err_disable_regulators;
111123278bf5SVinod Koul 	}
111223278bf5SVinod Koul 
111323278bf5SVinod Koul 	i2c_set_clientdata(client, lt9611);
111423278bf5SVinod Koul 
111523278bf5SVinod Koul 	lt9611->bridge.funcs = &lt9611_bridge_funcs;
111623278bf5SVinod Koul 	lt9611->bridge.of_node = client->dev.of_node;
111723278bf5SVinod Koul 	lt9611->bridge.ops = DRM_BRIDGE_OP_DETECT | DRM_BRIDGE_OP_EDID |
111823278bf5SVinod Koul 			     DRM_BRIDGE_OP_HPD | DRM_BRIDGE_OP_MODES;
111923278bf5SVinod Koul 	lt9611->bridge.type = DRM_MODE_CONNECTOR_HDMIA;
112023278bf5SVinod Koul 
112123278bf5SVinod Koul 	drm_bridge_add(&lt9611->bridge);
112223278bf5SVinod Koul 
1123fef604dbSMaxime Ripard 	/* Attach primary DSI */
1124fef604dbSMaxime Ripard 	lt9611->dsi0 = lt9611_attach_dsi(lt9611, lt9611->dsi0_node);
1125fef604dbSMaxime Ripard 	if (IS_ERR(lt9611->dsi0)) {
1126fef604dbSMaxime Ripard 		ret = PTR_ERR(lt9611->dsi0);
1127fef604dbSMaxime Ripard 		goto err_remove_bridge;
1128fef604dbSMaxime Ripard 	}
1129fef604dbSMaxime Ripard 
1130fef604dbSMaxime Ripard 	/* Attach secondary DSI, if specified */
1131fef604dbSMaxime Ripard 	if (lt9611->dsi1_node) {
1132fef604dbSMaxime Ripard 		lt9611->dsi1 = lt9611_attach_dsi(lt9611, lt9611->dsi1_node);
1133fef604dbSMaxime Ripard 		if (IS_ERR(lt9611->dsi1)) {
1134fef604dbSMaxime Ripard 			ret = PTR_ERR(lt9611->dsi1);
1135fef604dbSMaxime Ripard 			goto err_remove_bridge;
1136fef604dbSMaxime Ripard 		}
1137fef604dbSMaxime Ripard 	}
1138fef604dbSMaxime Ripard 
113923278bf5SVinod Koul 	lt9611_enable_hpd_interrupts(lt9611);
114023278bf5SVinod Koul 
11419987151aSChristophe JAILLET 	ret = lt9611_audio_init(dev, lt9611);
11429987151aSChristophe JAILLET 	if (ret)
11439987151aSChristophe JAILLET 		goto err_remove_bridge;
11449987151aSChristophe JAILLET 
11459987151aSChristophe JAILLET 	return 0;
114623278bf5SVinod Koul 
1147fef604dbSMaxime Ripard err_remove_bridge:
1148fef604dbSMaxime Ripard 	drm_bridge_remove(&lt9611->bridge);
1149fef604dbSMaxime Ripard 
115023278bf5SVinod Koul err_disable_regulators:
115123278bf5SVinod Koul 	regulator_bulk_disable(ARRAY_SIZE(lt9611->supplies), lt9611->supplies);
115223278bf5SVinod Koul 
115323278bf5SVinod Koul err_of_put:
115423278bf5SVinod Koul 	of_node_put(lt9611->dsi1_node);
115523278bf5SVinod Koul 	of_node_put(lt9611->dsi0_node);
115623278bf5SVinod Koul 
115723278bf5SVinod Koul 	return ret;
115823278bf5SVinod Koul }
115923278bf5SVinod Koul 
lt9611_remove(struct i2c_client * client)1160ed5c2f5fSUwe Kleine-König static void lt9611_remove(struct i2c_client *client)
116123278bf5SVinod Koul {
116223278bf5SVinod Koul 	struct lt9611 *lt9611 = i2c_get_clientdata(client);
116323278bf5SVinod Koul 
116423278bf5SVinod Koul 	disable_irq(client->irq);
116523278bf5SVinod Koul 	lt9611_audio_exit(lt9611);
116623278bf5SVinod Koul 	drm_bridge_remove(&lt9611->bridge);
116723278bf5SVinod Koul 
116823278bf5SVinod Koul 	regulator_bulk_disable(ARRAY_SIZE(lt9611->supplies), lt9611->supplies);
116923278bf5SVinod Koul 
117023278bf5SVinod Koul 	of_node_put(lt9611->dsi1_node);
117123278bf5SVinod Koul 	of_node_put(lt9611->dsi0_node);
117223278bf5SVinod Koul }
117323278bf5SVinod Koul 
117423278bf5SVinod Koul static struct i2c_device_id lt9611_id[] = {
117523278bf5SVinod Koul 	{ "lontium,lt9611", 0 },
117623278bf5SVinod Koul 	{}
117723278bf5SVinod Koul };
11788d0b1fe8SZou Wei MODULE_DEVICE_TABLE(i2c, lt9611_id);
117923278bf5SVinod Koul 
118023278bf5SVinod Koul static const struct of_device_id lt9611_match_table[] = {
118123278bf5SVinod Koul 	{ .compatible = "lontium,lt9611" },
118223278bf5SVinod Koul 	{ }
118323278bf5SVinod Koul };
118423278bf5SVinod Koul MODULE_DEVICE_TABLE(of, lt9611_match_table);
118523278bf5SVinod Koul 
118623278bf5SVinod Koul static struct i2c_driver lt9611_driver = {
118723278bf5SVinod Koul 	.driver = {
118823278bf5SVinod Koul 		.name = "lt9611",
118923278bf5SVinod Koul 		.of_match_table = lt9611_match_table,
119023278bf5SVinod Koul 	},
1191332af828SUwe Kleine-König 	.probe = lt9611_probe,
119223278bf5SVinod Koul 	.remove = lt9611_remove,
119323278bf5SVinod Koul 	.id_table = lt9611_id,
119423278bf5SVinod Koul };
119523278bf5SVinod Koul module_i2c_driver(lt9611_driver);
119623278bf5SVinod Koul 
119723278bf5SVinod Koul MODULE_LICENSE("GPL v2");
1198