xref: /OK3568_Linux_fs/kernel/drivers/gpu/drm/panel/panel-sharp-lq101r1sx01.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * Copyright (C) 2014 NVIDIA Corporation
4*4882a593Smuzhiyun  */
5*4882a593Smuzhiyun 
6*4882a593Smuzhiyun #include <linux/delay.h>
7*4882a593Smuzhiyun #include <linux/gpio/consumer.h>
8*4882a593Smuzhiyun #include <linux/module.h>
9*4882a593Smuzhiyun #include <linux/of.h>
10*4882a593Smuzhiyun #include <linux/regulator/consumer.h>
11*4882a593Smuzhiyun 
12*4882a593Smuzhiyun #include <video/mipi_display.h>
13*4882a593Smuzhiyun 
14*4882a593Smuzhiyun #include <drm/drm_crtc.h>
15*4882a593Smuzhiyun #include <drm/drm_device.h>
16*4882a593Smuzhiyun #include <drm/drm_mipi_dsi.h>
17*4882a593Smuzhiyun #include <drm/drm_panel.h>
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun struct sharp_panel {
20*4882a593Smuzhiyun 	struct drm_panel base;
21*4882a593Smuzhiyun 	/* the datasheet refers to them as DSI-LINK1 and DSI-LINK2 */
22*4882a593Smuzhiyun 	struct mipi_dsi_device *link1;
23*4882a593Smuzhiyun 	struct mipi_dsi_device *link2;
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun 	struct regulator *supply;
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun 	bool prepared;
28*4882a593Smuzhiyun 	bool enabled;
29*4882a593Smuzhiyun 
30*4882a593Smuzhiyun 	const struct drm_display_mode *mode;
31*4882a593Smuzhiyun };
32*4882a593Smuzhiyun 
to_sharp_panel(struct drm_panel * panel)33*4882a593Smuzhiyun static inline struct sharp_panel *to_sharp_panel(struct drm_panel *panel)
34*4882a593Smuzhiyun {
35*4882a593Smuzhiyun 	return container_of(panel, struct sharp_panel, base);
36*4882a593Smuzhiyun }
37*4882a593Smuzhiyun 
sharp_wait_frames(struct sharp_panel * sharp,unsigned int frames)38*4882a593Smuzhiyun static void sharp_wait_frames(struct sharp_panel *sharp, unsigned int frames)
39*4882a593Smuzhiyun {
40*4882a593Smuzhiyun 	unsigned int refresh = drm_mode_vrefresh(sharp->mode);
41*4882a593Smuzhiyun 
42*4882a593Smuzhiyun 	if (WARN_ON(frames > refresh))
43*4882a593Smuzhiyun 		return;
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun 	msleep(1000 / (refresh / frames));
46*4882a593Smuzhiyun }
47*4882a593Smuzhiyun 
sharp_panel_write(struct sharp_panel * sharp,u16 offset,u8 value)48*4882a593Smuzhiyun static int sharp_panel_write(struct sharp_panel *sharp, u16 offset, u8 value)
49*4882a593Smuzhiyun {
50*4882a593Smuzhiyun 	u8 payload[3] = { offset >> 8, offset & 0xff, value };
51*4882a593Smuzhiyun 	struct mipi_dsi_device *dsi = sharp->link1;
52*4882a593Smuzhiyun 	ssize_t err;
53*4882a593Smuzhiyun 
54*4882a593Smuzhiyun 	err = mipi_dsi_generic_write(dsi, payload, sizeof(payload));
55*4882a593Smuzhiyun 	if (err < 0) {
56*4882a593Smuzhiyun 		dev_err(&dsi->dev, "failed to write %02x to %04x: %zd\n",
57*4882a593Smuzhiyun 			value, offset, err);
58*4882a593Smuzhiyun 		return err;
59*4882a593Smuzhiyun 	}
60*4882a593Smuzhiyun 
61*4882a593Smuzhiyun 	err = mipi_dsi_dcs_nop(dsi);
62*4882a593Smuzhiyun 	if (err < 0) {
63*4882a593Smuzhiyun 		dev_err(&dsi->dev, "failed to send DCS nop: %zd\n", err);
64*4882a593Smuzhiyun 		return err;
65*4882a593Smuzhiyun 	}
66*4882a593Smuzhiyun 
67*4882a593Smuzhiyun 	usleep_range(10, 20);
68*4882a593Smuzhiyun 
69*4882a593Smuzhiyun 	return 0;
70*4882a593Smuzhiyun }
71*4882a593Smuzhiyun 
sharp_panel_read(struct sharp_panel * sharp,u16 offset,u8 * value)72*4882a593Smuzhiyun static __maybe_unused int sharp_panel_read(struct sharp_panel *sharp,
73*4882a593Smuzhiyun 					   u16 offset, u8 *value)
74*4882a593Smuzhiyun {
75*4882a593Smuzhiyun 	ssize_t err;
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun 	cpu_to_be16s(&offset);
78*4882a593Smuzhiyun 
79*4882a593Smuzhiyun 	err = mipi_dsi_generic_read(sharp->link1, &offset, sizeof(offset),
80*4882a593Smuzhiyun 				    value, sizeof(*value));
81*4882a593Smuzhiyun 	if (err < 0)
82*4882a593Smuzhiyun 		dev_err(&sharp->link1->dev, "failed to read from %04x: %zd\n",
83*4882a593Smuzhiyun 			offset, err);
84*4882a593Smuzhiyun 
85*4882a593Smuzhiyun 	return err;
86*4882a593Smuzhiyun }
87*4882a593Smuzhiyun 
sharp_panel_disable(struct drm_panel * panel)88*4882a593Smuzhiyun static int sharp_panel_disable(struct drm_panel *panel)
89*4882a593Smuzhiyun {
90*4882a593Smuzhiyun 	struct sharp_panel *sharp = to_sharp_panel(panel);
91*4882a593Smuzhiyun 
92*4882a593Smuzhiyun 	if (!sharp->enabled)
93*4882a593Smuzhiyun 		return 0;
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	sharp->enabled = false;
96*4882a593Smuzhiyun 
97*4882a593Smuzhiyun 	return 0;
98*4882a593Smuzhiyun }
99*4882a593Smuzhiyun 
sharp_panel_unprepare(struct drm_panel * panel)100*4882a593Smuzhiyun static int sharp_panel_unprepare(struct drm_panel *panel)
101*4882a593Smuzhiyun {
102*4882a593Smuzhiyun 	struct sharp_panel *sharp = to_sharp_panel(panel);
103*4882a593Smuzhiyun 	int err;
104*4882a593Smuzhiyun 
105*4882a593Smuzhiyun 	if (!sharp->prepared)
106*4882a593Smuzhiyun 		return 0;
107*4882a593Smuzhiyun 
108*4882a593Smuzhiyun 	sharp_wait_frames(sharp, 4);
109*4882a593Smuzhiyun 
110*4882a593Smuzhiyun 	err = mipi_dsi_dcs_set_display_off(sharp->link1);
111*4882a593Smuzhiyun 	if (err < 0)
112*4882a593Smuzhiyun 		dev_err(panel->dev, "failed to set display off: %d\n", err);
113*4882a593Smuzhiyun 
114*4882a593Smuzhiyun 	err = mipi_dsi_dcs_enter_sleep_mode(sharp->link1);
115*4882a593Smuzhiyun 	if (err < 0)
116*4882a593Smuzhiyun 		dev_err(panel->dev, "failed to enter sleep mode: %d\n", err);
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun 	msleep(120);
119*4882a593Smuzhiyun 
120*4882a593Smuzhiyun 	regulator_disable(sharp->supply);
121*4882a593Smuzhiyun 
122*4882a593Smuzhiyun 	sharp->prepared = false;
123*4882a593Smuzhiyun 
124*4882a593Smuzhiyun 	return 0;
125*4882a593Smuzhiyun }
126*4882a593Smuzhiyun 
sharp_setup_symmetrical_split(struct mipi_dsi_device * left,struct mipi_dsi_device * right,const struct drm_display_mode * mode)127*4882a593Smuzhiyun static int sharp_setup_symmetrical_split(struct mipi_dsi_device *left,
128*4882a593Smuzhiyun 					 struct mipi_dsi_device *right,
129*4882a593Smuzhiyun 					 const struct drm_display_mode *mode)
130*4882a593Smuzhiyun {
131*4882a593Smuzhiyun 	int err;
132*4882a593Smuzhiyun 
133*4882a593Smuzhiyun 	err = mipi_dsi_dcs_set_column_address(left, 0, mode->hdisplay / 2 - 1);
134*4882a593Smuzhiyun 	if (err < 0) {
135*4882a593Smuzhiyun 		dev_err(&left->dev, "failed to set column address: %d\n", err);
136*4882a593Smuzhiyun 		return err;
137*4882a593Smuzhiyun 	}
138*4882a593Smuzhiyun 
139*4882a593Smuzhiyun 	err = mipi_dsi_dcs_set_page_address(left, 0, mode->vdisplay - 1);
140*4882a593Smuzhiyun 	if (err < 0) {
141*4882a593Smuzhiyun 		dev_err(&left->dev, "failed to set page address: %d\n", err);
142*4882a593Smuzhiyun 		return err;
143*4882a593Smuzhiyun 	}
144*4882a593Smuzhiyun 
145*4882a593Smuzhiyun 	err = mipi_dsi_dcs_set_column_address(right, mode->hdisplay / 2,
146*4882a593Smuzhiyun 					      mode->hdisplay - 1);
147*4882a593Smuzhiyun 	if (err < 0) {
148*4882a593Smuzhiyun 		dev_err(&right->dev, "failed to set column address: %d\n", err);
149*4882a593Smuzhiyun 		return err;
150*4882a593Smuzhiyun 	}
151*4882a593Smuzhiyun 
152*4882a593Smuzhiyun 	err = mipi_dsi_dcs_set_page_address(right, 0, mode->vdisplay - 1);
153*4882a593Smuzhiyun 	if (err < 0) {
154*4882a593Smuzhiyun 		dev_err(&right->dev, "failed to set page address: %d\n", err);
155*4882a593Smuzhiyun 		return err;
156*4882a593Smuzhiyun 	}
157*4882a593Smuzhiyun 
158*4882a593Smuzhiyun 	return 0;
159*4882a593Smuzhiyun }
160*4882a593Smuzhiyun 
sharp_panel_prepare(struct drm_panel * panel)161*4882a593Smuzhiyun static int sharp_panel_prepare(struct drm_panel *panel)
162*4882a593Smuzhiyun {
163*4882a593Smuzhiyun 	struct sharp_panel *sharp = to_sharp_panel(panel);
164*4882a593Smuzhiyun 	u8 format = MIPI_DCS_PIXEL_FMT_24BIT;
165*4882a593Smuzhiyun 	int err;
166*4882a593Smuzhiyun 
167*4882a593Smuzhiyun 	if (sharp->prepared)
168*4882a593Smuzhiyun 		return 0;
169*4882a593Smuzhiyun 
170*4882a593Smuzhiyun 	err = regulator_enable(sharp->supply);
171*4882a593Smuzhiyun 	if (err < 0)
172*4882a593Smuzhiyun 		return err;
173*4882a593Smuzhiyun 
174*4882a593Smuzhiyun 	/*
175*4882a593Smuzhiyun 	 * According to the datasheet, the panel needs around 10 ms to fully
176*4882a593Smuzhiyun 	 * power up. At least another 120 ms is required before exiting sleep
177*4882a593Smuzhiyun 	 * mode to make sure the panel is ready. Throw in another 20 ms for
178*4882a593Smuzhiyun 	 * good measure.
179*4882a593Smuzhiyun 	 */
180*4882a593Smuzhiyun 	msleep(150);
181*4882a593Smuzhiyun 
182*4882a593Smuzhiyun 	err = mipi_dsi_dcs_exit_sleep_mode(sharp->link1);
183*4882a593Smuzhiyun 	if (err < 0) {
184*4882a593Smuzhiyun 		dev_err(panel->dev, "failed to exit sleep mode: %d\n", err);
185*4882a593Smuzhiyun 		goto poweroff;
186*4882a593Smuzhiyun 	}
187*4882a593Smuzhiyun 
188*4882a593Smuzhiyun 	/*
189*4882a593Smuzhiyun 	 * The MIPI DCS specification mandates this delay only between the
190*4882a593Smuzhiyun 	 * exit_sleep_mode and enter_sleep_mode commands, so it isn't strictly
191*4882a593Smuzhiyun 	 * necessary here.
192*4882a593Smuzhiyun 	 */
193*4882a593Smuzhiyun 	/*
194*4882a593Smuzhiyun 	msleep(120);
195*4882a593Smuzhiyun 	*/
196*4882a593Smuzhiyun 
197*4882a593Smuzhiyun 	/* set left-right mode */
198*4882a593Smuzhiyun 	err = sharp_panel_write(sharp, 0x1000, 0x2a);
199*4882a593Smuzhiyun 	if (err < 0) {
200*4882a593Smuzhiyun 		dev_err(panel->dev, "failed to set left-right mode: %d\n", err);
201*4882a593Smuzhiyun 		goto poweroff;
202*4882a593Smuzhiyun 	}
203*4882a593Smuzhiyun 
204*4882a593Smuzhiyun 	/* enable command mode */
205*4882a593Smuzhiyun 	err = sharp_panel_write(sharp, 0x1001, 0x01);
206*4882a593Smuzhiyun 	if (err < 0) {
207*4882a593Smuzhiyun 		dev_err(panel->dev, "failed to enable command mode: %d\n", err);
208*4882a593Smuzhiyun 		goto poweroff;
209*4882a593Smuzhiyun 	}
210*4882a593Smuzhiyun 
211*4882a593Smuzhiyun 	err = mipi_dsi_dcs_set_pixel_format(sharp->link1, format);
212*4882a593Smuzhiyun 	if (err < 0) {
213*4882a593Smuzhiyun 		dev_err(panel->dev, "failed to set pixel format: %d\n", err);
214*4882a593Smuzhiyun 		goto poweroff;
215*4882a593Smuzhiyun 	}
216*4882a593Smuzhiyun 
217*4882a593Smuzhiyun 	/*
218*4882a593Smuzhiyun 	 * TODO: The device supports both left-right and even-odd split
219*4882a593Smuzhiyun 	 * configurations, but this driver currently supports only the left-
220*4882a593Smuzhiyun 	 * right split. To support a different mode a mechanism needs to be
221*4882a593Smuzhiyun 	 * put in place to communicate the configuration back to the DSI host
222*4882a593Smuzhiyun 	 * controller.
223*4882a593Smuzhiyun 	 */
224*4882a593Smuzhiyun 	err = sharp_setup_symmetrical_split(sharp->link1, sharp->link2,
225*4882a593Smuzhiyun 					    sharp->mode);
226*4882a593Smuzhiyun 	if (err < 0) {
227*4882a593Smuzhiyun 		dev_err(panel->dev, "failed to set up symmetrical split: %d\n",
228*4882a593Smuzhiyun 			err);
229*4882a593Smuzhiyun 		goto poweroff;
230*4882a593Smuzhiyun 	}
231*4882a593Smuzhiyun 
232*4882a593Smuzhiyun 	err = mipi_dsi_dcs_set_display_on(sharp->link1);
233*4882a593Smuzhiyun 	if (err < 0) {
234*4882a593Smuzhiyun 		dev_err(panel->dev, "failed to set display on: %d\n", err);
235*4882a593Smuzhiyun 		goto poweroff;
236*4882a593Smuzhiyun 	}
237*4882a593Smuzhiyun 
238*4882a593Smuzhiyun 	sharp->prepared = true;
239*4882a593Smuzhiyun 
240*4882a593Smuzhiyun 	/* wait for 6 frames before continuing */
241*4882a593Smuzhiyun 	sharp_wait_frames(sharp, 6);
242*4882a593Smuzhiyun 
243*4882a593Smuzhiyun 	return 0;
244*4882a593Smuzhiyun 
245*4882a593Smuzhiyun poweroff:
246*4882a593Smuzhiyun 	regulator_disable(sharp->supply);
247*4882a593Smuzhiyun 	return err;
248*4882a593Smuzhiyun }
249*4882a593Smuzhiyun 
sharp_panel_enable(struct drm_panel * panel)250*4882a593Smuzhiyun static int sharp_panel_enable(struct drm_panel *panel)
251*4882a593Smuzhiyun {
252*4882a593Smuzhiyun 	struct sharp_panel *sharp = to_sharp_panel(panel);
253*4882a593Smuzhiyun 
254*4882a593Smuzhiyun 	if (sharp->enabled)
255*4882a593Smuzhiyun 		return 0;
256*4882a593Smuzhiyun 
257*4882a593Smuzhiyun 	sharp->enabled = true;
258*4882a593Smuzhiyun 
259*4882a593Smuzhiyun 	return 0;
260*4882a593Smuzhiyun }
261*4882a593Smuzhiyun 
262*4882a593Smuzhiyun static const struct drm_display_mode default_mode = {
263*4882a593Smuzhiyun 	.clock = 278000,
264*4882a593Smuzhiyun 	.hdisplay = 2560,
265*4882a593Smuzhiyun 	.hsync_start = 2560 + 128,
266*4882a593Smuzhiyun 	.hsync_end = 2560 + 128 + 64,
267*4882a593Smuzhiyun 	.htotal = 2560 + 128 + 64 + 64,
268*4882a593Smuzhiyun 	.vdisplay = 1600,
269*4882a593Smuzhiyun 	.vsync_start = 1600 + 4,
270*4882a593Smuzhiyun 	.vsync_end = 1600 + 4 + 8,
271*4882a593Smuzhiyun 	.vtotal = 1600 + 4 + 8 + 32,
272*4882a593Smuzhiyun };
273*4882a593Smuzhiyun 
sharp_panel_get_modes(struct drm_panel * panel,struct drm_connector * connector)274*4882a593Smuzhiyun static int sharp_panel_get_modes(struct drm_panel *panel,
275*4882a593Smuzhiyun 				 struct drm_connector *connector)
276*4882a593Smuzhiyun {
277*4882a593Smuzhiyun 	struct drm_display_mode *mode;
278*4882a593Smuzhiyun 
279*4882a593Smuzhiyun 	mode = drm_mode_duplicate(connector->dev, &default_mode);
280*4882a593Smuzhiyun 	if (!mode) {
281*4882a593Smuzhiyun 		dev_err(panel->dev, "failed to add mode %ux%ux@%u\n",
282*4882a593Smuzhiyun 			default_mode.hdisplay, default_mode.vdisplay,
283*4882a593Smuzhiyun 			drm_mode_vrefresh(&default_mode));
284*4882a593Smuzhiyun 		return -ENOMEM;
285*4882a593Smuzhiyun 	}
286*4882a593Smuzhiyun 
287*4882a593Smuzhiyun 	drm_mode_set_name(mode);
288*4882a593Smuzhiyun 
289*4882a593Smuzhiyun 	drm_mode_probed_add(connector, mode);
290*4882a593Smuzhiyun 
291*4882a593Smuzhiyun 	connector->display_info.width_mm = 217;
292*4882a593Smuzhiyun 	connector->display_info.height_mm = 136;
293*4882a593Smuzhiyun 
294*4882a593Smuzhiyun 	return 1;
295*4882a593Smuzhiyun }
296*4882a593Smuzhiyun 
297*4882a593Smuzhiyun static const struct drm_panel_funcs sharp_panel_funcs = {
298*4882a593Smuzhiyun 	.disable = sharp_panel_disable,
299*4882a593Smuzhiyun 	.unprepare = sharp_panel_unprepare,
300*4882a593Smuzhiyun 	.prepare = sharp_panel_prepare,
301*4882a593Smuzhiyun 	.enable = sharp_panel_enable,
302*4882a593Smuzhiyun 	.get_modes = sharp_panel_get_modes,
303*4882a593Smuzhiyun };
304*4882a593Smuzhiyun 
305*4882a593Smuzhiyun static const struct of_device_id sharp_of_match[] = {
306*4882a593Smuzhiyun 	{ .compatible = "sharp,lq101r1sx01", },
307*4882a593Smuzhiyun 	{ }
308*4882a593Smuzhiyun };
309*4882a593Smuzhiyun MODULE_DEVICE_TABLE(of, sharp_of_match);
310*4882a593Smuzhiyun 
sharp_panel_add(struct sharp_panel * sharp)311*4882a593Smuzhiyun static int sharp_panel_add(struct sharp_panel *sharp)
312*4882a593Smuzhiyun {
313*4882a593Smuzhiyun 	int ret;
314*4882a593Smuzhiyun 
315*4882a593Smuzhiyun 	sharp->mode = &default_mode;
316*4882a593Smuzhiyun 
317*4882a593Smuzhiyun 	sharp->supply = devm_regulator_get(&sharp->link1->dev, "power");
318*4882a593Smuzhiyun 	if (IS_ERR(sharp->supply))
319*4882a593Smuzhiyun 		return PTR_ERR(sharp->supply);
320*4882a593Smuzhiyun 
321*4882a593Smuzhiyun 	drm_panel_init(&sharp->base, &sharp->link1->dev, &sharp_panel_funcs,
322*4882a593Smuzhiyun 		       DRM_MODE_CONNECTOR_DSI);
323*4882a593Smuzhiyun 
324*4882a593Smuzhiyun 	ret = drm_panel_of_backlight(&sharp->base);
325*4882a593Smuzhiyun 	if (ret)
326*4882a593Smuzhiyun 		return ret;
327*4882a593Smuzhiyun 
328*4882a593Smuzhiyun 	drm_panel_add(&sharp->base);
329*4882a593Smuzhiyun 
330*4882a593Smuzhiyun 	return 0;
331*4882a593Smuzhiyun }
332*4882a593Smuzhiyun 
sharp_panel_del(struct sharp_panel * sharp)333*4882a593Smuzhiyun static void sharp_panel_del(struct sharp_panel *sharp)
334*4882a593Smuzhiyun {
335*4882a593Smuzhiyun 	if (sharp->base.dev)
336*4882a593Smuzhiyun 		drm_panel_remove(&sharp->base);
337*4882a593Smuzhiyun 
338*4882a593Smuzhiyun 	if (sharp->link2)
339*4882a593Smuzhiyun 		put_device(&sharp->link2->dev);
340*4882a593Smuzhiyun }
341*4882a593Smuzhiyun 
sharp_panel_probe(struct mipi_dsi_device * dsi)342*4882a593Smuzhiyun static int sharp_panel_probe(struct mipi_dsi_device *dsi)
343*4882a593Smuzhiyun {
344*4882a593Smuzhiyun 	struct mipi_dsi_device *secondary = NULL;
345*4882a593Smuzhiyun 	struct sharp_panel *sharp;
346*4882a593Smuzhiyun 	struct device_node *np;
347*4882a593Smuzhiyun 	int err;
348*4882a593Smuzhiyun 
349*4882a593Smuzhiyun 	dsi->lanes = 4;
350*4882a593Smuzhiyun 	dsi->format = MIPI_DSI_FMT_RGB888;
351*4882a593Smuzhiyun 	dsi->mode_flags = MIPI_DSI_MODE_LPM;
352*4882a593Smuzhiyun 
353*4882a593Smuzhiyun 	/* Find DSI-LINK1 */
354*4882a593Smuzhiyun 	np = of_parse_phandle(dsi->dev.of_node, "link2", 0);
355*4882a593Smuzhiyun 	if (np) {
356*4882a593Smuzhiyun 		secondary = of_find_mipi_dsi_device_by_node(np);
357*4882a593Smuzhiyun 		of_node_put(np);
358*4882a593Smuzhiyun 
359*4882a593Smuzhiyun 		if (!secondary)
360*4882a593Smuzhiyun 			return -EPROBE_DEFER;
361*4882a593Smuzhiyun 	}
362*4882a593Smuzhiyun 
363*4882a593Smuzhiyun 	/* register a panel for only the DSI-LINK1 interface */
364*4882a593Smuzhiyun 	if (secondary) {
365*4882a593Smuzhiyun 		sharp = devm_kzalloc(&dsi->dev, sizeof(*sharp), GFP_KERNEL);
366*4882a593Smuzhiyun 		if (!sharp) {
367*4882a593Smuzhiyun 			put_device(&secondary->dev);
368*4882a593Smuzhiyun 			return -ENOMEM;
369*4882a593Smuzhiyun 		}
370*4882a593Smuzhiyun 
371*4882a593Smuzhiyun 		mipi_dsi_set_drvdata(dsi, sharp);
372*4882a593Smuzhiyun 
373*4882a593Smuzhiyun 		sharp->link2 = secondary;
374*4882a593Smuzhiyun 		sharp->link1 = dsi;
375*4882a593Smuzhiyun 
376*4882a593Smuzhiyun 		err = sharp_panel_add(sharp);
377*4882a593Smuzhiyun 		if (err < 0) {
378*4882a593Smuzhiyun 			put_device(&secondary->dev);
379*4882a593Smuzhiyun 			return err;
380*4882a593Smuzhiyun 		}
381*4882a593Smuzhiyun 	}
382*4882a593Smuzhiyun 
383*4882a593Smuzhiyun 	err = mipi_dsi_attach(dsi);
384*4882a593Smuzhiyun 	if (err < 0) {
385*4882a593Smuzhiyun 		if (secondary)
386*4882a593Smuzhiyun 			sharp_panel_del(sharp);
387*4882a593Smuzhiyun 
388*4882a593Smuzhiyun 		return err;
389*4882a593Smuzhiyun 	}
390*4882a593Smuzhiyun 
391*4882a593Smuzhiyun 	return 0;
392*4882a593Smuzhiyun }
393*4882a593Smuzhiyun 
sharp_panel_remove(struct mipi_dsi_device * dsi)394*4882a593Smuzhiyun static int sharp_panel_remove(struct mipi_dsi_device *dsi)
395*4882a593Smuzhiyun {
396*4882a593Smuzhiyun 	struct sharp_panel *sharp = mipi_dsi_get_drvdata(dsi);
397*4882a593Smuzhiyun 	int err;
398*4882a593Smuzhiyun 
399*4882a593Smuzhiyun 	/* only detach from host for the DSI-LINK2 interface */
400*4882a593Smuzhiyun 	if (!sharp) {
401*4882a593Smuzhiyun 		mipi_dsi_detach(dsi);
402*4882a593Smuzhiyun 		return 0;
403*4882a593Smuzhiyun 	}
404*4882a593Smuzhiyun 
405*4882a593Smuzhiyun 	err = drm_panel_disable(&sharp->base);
406*4882a593Smuzhiyun 	if (err < 0)
407*4882a593Smuzhiyun 		dev_err(&dsi->dev, "failed to disable panel: %d\n", err);
408*4882a593Smuzhiyun 
409*4882a593Smuzhiyun 	err = mipi_dsi_detach(dsi);
410*4882a593Smuzhiyun 	if (err < 0)
411*4882a593Smuzhiyun 		dev_err(&dsi->dev, "failed to detach from DSI host: %d\n", err);
412*4882a593Smuzhiyun 
413*4882a593Smuzhiyun 	sharp_panel_del(sharp);
414*4882a593Smuzhiyun 
415*4882a593Smuzhiyun 	return 0;
416*4882a593Smuzhiyun }
417*4882a593Smuzhiyun 
sharp_panel_shutdown(struct mipi_dsi_device * dsi)418*4882a593Smuzhiyun static void sharp_panel_shutdown(struct mipi_dsi_device *dsi)
419*4882a593Smuzhiyun {
420*4882a593Smuzhiyun 	struct sharp_panel *sharp = mipi_dsi_get_drvdata(dsi);
421*4882a593Smuzhiyun 
422*4882a593Smuzhiyun 	/* nothing to do for DSI-LINK2 */
423*4882a593Smuzhiyun 	if (!sharp)
424*4882a593Smuzhiyun 		return;
425*4882a593Smuzhiyun 
426*4882a593Smuzhiyun 	drm_panel_disable(&sharp->base);
427*4882a593Smuzhiyun }
428*4882a593Smuzhiyun 
429*4882a593Smuzhiyun static struct mipi_dsi_driver sharp_panel_driver = {
430*4882a593Smuzhiyun 	.driver = {
431*4882a593Smuzhiyun 		.name = "panel-sharp-lq101r1sx01",
432*4882a593Smuzhiyun 		.of_match_table = sharp_of_match,
433*4882a593Smuzhiyun 	},
434*4882a593Smuzhiyun 	.probe = sharp_panel_probe,
435*4882a593Smuzhiyun 	.remove = sharp_panel_remove,
436*4882a593Smuzhiyun 	.shutdown = sharp_panel_shutdown,
437*4882a593Smuzhiyun };
438*4882a593Smuzhiyun module_mipi_dsi_driver(sharp_panel_driver);
439*4882a593Smuzhiyun 
440*4882a593Smuzhiyun MODULE_AUTHOR("Thierry Reding <treding@nvidia.com>");
441*4882a593Smuzhiyun MODULE_DESCRIPTION("Sharp LQ101R1SX01 panel driver");
442*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
443