1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (C) 2012 Texas Instruments
4 * Author: Rob Clark <robdclark@gmail.com>
7 #include <linux/backlight.h>
8 #include <linux/gpio/consumer.h>
9 #include <linux/platform_device.h>
11 #include <video/display_timing.h>
12 #include <video/of_display_timing.h>
13 #include <video/videomode.h>
15 #include <drm/drm_atomic_state_helper.h>
16 #include <drm/drm_connector.h>
17 #include <drm/drm_modeset_helper_vtables.h>
18 #include <drm/drm_probe_helper.h>
19 #include <drm/drm_simple_kms_helper.h>
21 #include "tilcdc_drv.h"
22 #include "tilcdc_panel.h"
25 struct tilcdc_module base
;
26 struct tilcdc_panel_info
*info
;
27 struct display_timings
*timings
;
28 struct backlight_device
*backlight
;
29 struct gpio_desc
*enable_gpio
;
31 #define to_panel_module(x) container_of(x, struct panel_module, base)
38 struct panel_encoder
{
39 struct drm_encoder base
;
40 struct panel_module
*mod
;
42 #define to_panel_encoder(x) container_of(x, struct panel_encoder, base)
44 static void panel_encoder_dpms(struct drm_encoder
*encoder
, int mode
)
46 struct panel_encoder
*panel_encoder
= to_panel_encoder(encoder
);
47 struct backlight_device
*backlight
= panel_encoder
->mod
->backlight
;
48 struct gpio_desc
*gpio
= panel_encoder
->mod
->enable_gpio
;
51 backlight
->props
.power
= mode
== DRM_MODE_DPMS_ON
?
52 BACKLIGHT_POWER_ON
: BACKLIGHT_POWER_OFF
;
53 backlight_update_status(backlight
);
57 gpiod_set_value_cansleep(gpio
,
58 mode
== DRM_MODE_DPMS_ON
? 1 : 0);
61 static void panel_encoder_prepare(struct drm_encoder
*encoder
)
63 panel_encoder_dpms(encoder
, DRM_MODE_DPMS_OFF
);
66 static void panel_encoder_commit(struct drm_encoder
*encoder
)
68 panel_encoder_dpms(encoder
, DRM_MODE_DPMS_ON
);
71 static void panel_encoder_mode_set(struct drm_encoder
*encoder
,
72 struct drm_display_mode
*mode
,
73 struct drm_display_mode
*adjusted_mode
)
78 static const struct drm_encoder_helper_funcs panel_encoder_helper_funcs
= {
79 .dpms
= panel_encoder_dpms
,
80 .prepare
= panel_encoder_prepare
,
81 .commit
= panel_encoder_commit
,
82 .mode_set
= panel_encoder_mode_set
,
85 static struct drm_encoder
*panel_encoder_create(struct drm_device
*dev
,
86 struct panel_module
*mod
)
88 struct panel_encoder
*panel_encoder
;
89 struct drm_encoder
*encoder
;
92 panel_encoder
= devm_kzalloc(dev
->dev
, sizeof(*panel_encoder
),
97 panel_encoder
->mod
= mod
;
99 encoder
= &panel_encoder
->base
;
100 encoder
->possible_crtcs
= 1;
102 ret
= drm_simple_encoder_init(dev
, encoder
, DRM_MODE_ENCODER_LVDS
);
106 drm_encoder_helper_add(encoder
, &panel_encoder_helper_funcs
);
111 drm_encoder_cleanup(encoder
);
119 struct panel_connector
{
120 struct drm_connector base
;
122 struct drm_encoder
*encoder
; /* our connected encoder */
123 struct panel_module
*mod
;
125 #define to_panel_connector(x) container_of(x, struct panel_connector, base)
128 static void panel_connector_destroy(struct drm_connector
*connector
)
130 drm_connector_unregister(connector
);
131 drm_connector_cleanup(connector
);
134 static int panel_connector_get_modes(struct drm_connector
*connector
)
136 struct drm_device
*dev
= connector
->dev
;
137 struct panel_connector
*panel_connector
= to_panel_connector(connector
);
138 struct display_timings
*timings
= panel_connector
->mod
->timings
;
141 for (i
= 0; i
< timings
->num_timings
; i
++) {
142 struct drm_display_mode
*mode
;
145 if (videomode_from_timings(timings
, &vm
, i
))
148 mode
= drm_mode_create(dev
);
152 drm_display_mode_from_videomode(&vm
, mode
);
154 mode
->type
= DRM_MODE_TYPE_DRIVER
;
156 if (timings
->native_mode
== i
)
157 mode
->type
|= DRM_MODE_TYPE_PREFERRED
;
159 drm_mode_set_name(mode
);
160 drm_mode_probed_add(connector
, mode
);
166 static struct drm_encoder
*panel_connector_best_encoder(
167 struct drm_connector
*connector
)
169 struct panel_connector
*panel_connector
= to_panel_connector(connector
);
170 return panel_connector
->encoder
;
173 static const struct drm_connector_funcs panel_connector_funcs
= {
174 .destroy
= panel_connector_destroy
,
175 .fill_modes
= drm_helper_probe_single_connector_modes
,
176 .reset
= drm_atomic_helper_connector_reset
,
177 .atomic_duplicate_state
= drm_atomic_helper_connector_duplicate_state
,
178 .atomic_destroy_state
= drm_atomic_helper_connector_destroy_state
,
181 static const struct drm_connector_helper_funcs panel_connector_helper_funcs
= {
182 .get_modes
= panel_connector_get_modes
,
183 .best_encoder
= panel_connector_best_encoder
,
186 static struct drm_connector
*panel_connector_create(struct drm_device
*dev
,
187 struct panel_module
*mod
, struct drm_encoder
*encoder
)
189 struct panel_connector
*panel_connector
;
190 struct drm_connector
*connector
;
193 panel_connector
= devm_kzalloc(dev
->dev
, sizeof(*panel_connector
),
195 if (!panel_connector
)
198 panel_connector
->encoder
= encoder
;
199 panel_connector
->mod
= mod
;
201 connector
= &panel_connector
->base
;
203 drm_connector_init(dev
, connector
, &panel_connector_funcs
,
204 DRM_MODE_CONNECTOR_LVDS
);
205 drm_connector_helper_add(connector
, &panel_connector_helper_funcs
);
207 connector
->interlace_allowed
= 0;
208 connector
->doublescan_allowed
= 0;
210 ret
= drm_connector_attach_encoder(connector
, encoder
);
217 panel_connector_destroy(connector
);
225 static int panel_modeset_init(struct tilcdc_module
*mod
, struct drm_device
*dev
)
227 struct panel_module
*panel_mod
= to_panel_module(mod
);
228 struct tilcdc_drm_private
*priv
= dev
->dev_private
;
229 struct drm_encoder
*encoder
;
230 struct drm_connector
*connector
;
232 encoder
= panel_encoder_create(dev
, panel_mod
);
236 connector
= panel_connector_create(dev
, panel_mod
, encoder
);
240 priv
->encoders
[priv
->num_encoders
++] = encoder
;
241 priv
->connectors
[priv
->num_connectors
++] = connector
;
243 tilcdc_crtc_set_panel_info(priv
->crtc
,
244 to_panel_encoder(encoder
)->mod
->info
);
249 static const struct tilcdc_module_ops panel_module_ops
= {
250 .modeset_init
= panel_modeset_init
,
257 /* maybe move this somewhere common if it is needed by other outputs? */
258 static struct tilcdc_panel_info
*of_get_panel_info(struct device_node
*np
)
260 struct device_node
*info_np
;
261 struct tilcdc_panel_info
*info
;
265 pr_err("%s: no devicenode given\n", __func__
);
269 info_np
= of_get_child_by_name(np
, "panel-info");
271 pr_err("%s: could not find panel-info node\n", __func__
);
275 info
= kzalloc(sizeof(*info
), GFP_KERNEL
);
279 ret
|= of_property_read_u32(info_np
, "ac-bias", &info
->ac_bias
);
280 ret
|= of_property_read_u32(info_np
, "ac-bias-intrpt", &info
->ac_bias_intrpt
);
281 ret
|= of_property_read_u32(info_np
, "dma-burst-sz", &info
->dma_burst_sz
);
282 ret
|= of_property_read_u32(info_np
, "bpp", &info
->bpp
);
283 ret
|= of_property_read_u32(info_np
, "fdd", &info
->fdd
);
284 ret
|= of_property_read_u32(info_np
, "sync-edge", &info
->sync_edge
);
285 ret
|= of_property_read_u32(info_np
, "sync-ctrl", &info
->sync_ctrl
);
286 ret
|= of_property_read_u32(info_np
, "raster-order", &info
->raster_order
);
287 ret
|= of_property_read_u32(info_np
, "fifo-th", &info
->fifo_th
);
290 info
->tft_alt_mode
= of_property_read_bool(info_np
, "tft-alt-mode");
291 info
->invert_pxl_clk
= of_property_read_bool(info_np
, "invert-pxl-clk");
294 pr_err("%s: error reading panel-info properties\n", __func__
);
300 of_node_put(info_np
);
304 static int panel_probe(struct platform_device
*pdev
)
306 struct device_node
*node
= pdev
->dev
.of_node
;
307 struct backlight_device
*backlight
;
308 struct panel_module
*panel_mod
;
309 struct tilcdc_module
*mod
;
312 /* bail out early if no DT data: */
314 dev_err(&pdev
->dev
, "device-tree data is missing\n");
318 panel_mod
= devm_kzalloc(&pdev
->dev
, sizeof(*panel_mod
), GFP_KERNEL
);
322 backlight
= devm_of_find_backlight(&pdev
->dev
);
323 if (IS_ERR(backlight
))
324 return PTR_ERR(backlight
);
325 panel_mod
->backlight
= backlight
;
327 panel_mod
->enable_gpio
= devm_gpiod_get_optional(&pdev
->dev
, "enable",
329 if (IS_ERR(panel_mod
->enable_gpio
)) {
330 ret
= PTR_ERR(panel_mod
->enable_gpio
);
331 dev_err(&pdev
->dev
, "failed to request enable GPIO\n");
335 if (panel_mod
->enable_gpio
)
336 dev_info(&pdev
->dev
, "found enable GPIO\n");
338 mod
= &panel_mod
->base
;
339 pdev
->dev
.platform_data
= mod
;
341 tilcdc_module_init(mod
, "panel", &panel_module_ops
);
343 panel_mod
->timings
= of_get_display_timings(node
);
344 if (!panel_mod
->timings
) {
345 dev_err(&pdev
->dev
, "could not get panel timings\n");
350 panel_mod
->info
= of_get_panel_info(node
);
351 if (!panel_mod
->info
) {
352 dev_err(&pdev
->dev
, "could not get panel info\n");
360 display_timings_release(panel_mod
->timings
);
363 tilcdc_module_cleanup(mod
);
366 if (panel_mod
->backlight
)
367 put_device(&panel_mod
->backlight
->dev
);
371 static void panel_remove(struct platform_device
*pdev
)
373 struct tilcdc_module
*mod
= dev_get_platdata(&pdev
->dev
);
374 struct panel_module
*panel_mod
= to_panel_module(mod
);
375 struct backlight_device
*backlight
= panel_mod
->backlight
;
378 put_device(&backlight
->dev
);
380 display_timings_release(panel_mod
->timings
);
382 tilcdc_module_cleanup(mod
);
383 kfree(panel_mod
->info
);
386 static const struct of_device_id panel_of_match
[] = {
387 { .compatible
= "ti,tilcdc,panel", },
391 static struct platform_driver panel_driver
= {
392 .probe
= panel_probe
,
393 .remove
= panel_remove
,
395 .name
= "tilcdc-panel",
396 .of_match_table
= panel_of_match
,
400 int __init
tilcdc_panel_init(void)
402 return platform_driver_register(&panel_driver
);
405 void __exit
tilcdc_panel_fini(void)
407 platform_driver_unregister(&panel_driver
);