2 * Copyright (C) 2016 Laurent Pinchart <laurent.pinchart@ideasonboard.com>
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License as
6 * published by the Free Software Foundation; either version 2 of
7 * the License, or (at your option) any later version.
11 #include <drm/drm_bridge.h>
12 #include <drm/drm_panel.h>
14 #include <linux/gpio/consumer.h>
15 #include <linux/of_graph.h>
18 struct drm_bridge bridge
;
19 struct drm_bridge
*panel_bridge
;
20 struct gpio_desc
*powerdown_gpio
;
23 static int lvds_encoder_attach(struct drm_bridge
*bridge
)
25 struct lvds_encoder
*lvds_encoder
= container_of(bridge
,
29 return drm_bridge_attach(bridge
->encoder
, lvds_encoder
->panel_bridge
,
33 static void lvds_encoder_enable(struct drm_bridge
*bridge
)
35 struct lvds_encoder
*lvds_encoder
= container_of(bridge
,
39 if (lvds_encoder
->powerdown_gpio
)
40 gpiod_set_value_cansleep(lvds_encoder
->powerdown_gpio
, 0);
43 static void lvds_encoder_disable(struct drm_bridge
*bridge
)
45 struct lvds_encoder
*lvds_encoder
= container_of(bridge
,
49 if (lvds_encoder
->powerdown_gpio
)
50 gpiod_set_value_cansleep(lvds_encoder
->powerdown_gpio
, 1);
53 static struct drm_bridge_funcs funcs
= {
54 .attach
= lvds_encoder_attach
,
55 .enable
= lvds_encoder_enable
,
56 .disable
= lvds_encoder_disable
,
59 static int lvds_encoder_probe(struct platform_device
*pdev
)
61 struct device
*dev
= &pdev
->dev
;
62 struct device_node
*port
;
63 struct device_node
*endpoint
;
64 struct device_node
*panel_node
;
65 struct drm_panel
*panel
;
66 struct lvds_encoder
*lvds_encoder
;
68 lvds_encoder
= devm_kzalloc(dev
, sizeof(*lvds_encoder
), GFP_KERNEL
);
72 lvds_encoder
->powerdown_gpio
= devm_gpiod_get_optional(dev
, "powerdown",
74 if (IS_ERR(lvds_encoder
->powerdown_gpio
)) {
75 int err
= PTR_ERR(lvds_encoder
->powerdown_gpio
);
77 if (err
!= -EPROBE_DEFER
)
78 dev_err(dev
, "powerdown GPIO failure: %d\n", err
);
82 /* Locate the panel DT node. */
83 port
= of_graph_get_port_by_id(dev
->of_node
, 1);
85 dev_dbg(dev
, "port 1 not found\n");
89 endpoint
= of_get_child_by_name(port
, "endpoint");
92 dev_dbg(dev
, "no endpoint for port 1\n");
96 panel_node
= of_graph_get_remote_port_parent(endpoint
);
97 of_node_put(endpoint
);
99 dev_dbg(dev
, "no remote endpoint for port 1\n");
103 panel
= of_drm_find_panel(panel_node
);
104 of_node_put(panel_node
);
106 dev_dbg(dev
, "panel not found, deferring probe\n");
107 return PTR_ERR(panel
);
110 lvds_encoder
->panel_bridge
=
111 devm_drm_panel_bridge_add(dev
, panel
, DRM_MODE_CONNECTOR_LVDS
);
112 if (IS_ERR(lvds_encoder
->panel_bridge
))
113 return PTR_ERR(lvds_encoder
->panel_bridge
);
115 /* The panel_bridge bridge is attached to the panel's of_node,
116 * but we need a bridge attached to our of_node for our user
119 lvds_encoder
->bridge
.of_node
= dev
->of_node
;
120 lvds_encoder
->bridge
.funcs
= &funcs
;
121 drm_bridge_add(&lvds_encoder
->bridge
);
123 platform_set_drvdata(pdev
, lvds_encoder
);
128 static int lvds_encoder_remove(struct platform_device
*pdev
)
130 struct lvds_encoder
*lvds_encoder
= platform_get_drvdata(pdev
);
132 drm_bridge_remove(&lvds_encoder
->bridge
);
137 static const struct of_device_id lvds_encoder_match
[] = {
138 { .compatible
= "lvds-encoder" },
139 { .compatible
= "thine,thc63lvdm83d" },
142 MODULE_DEVICE_TABLE(of
, lvds_encoder_match
);
144 static struct platform_driver lvds_encoder_driver
= {
145 .probe
= lvds_encoder_probe
,
146 .remove
= lvds_encoder_remove
,
148 .name
= "lvds-encoder",
149 .of_match_table
= lvds_encoder_match
,
152 module_platform_driver(lvds_encoder_driver
);
154 MODULE_AUTHOR("Laurent Pinchart <laurent.pinchart@ideasonboard.com>");
155 MODULE_DESCRIPTION("Transparent parallel to LVDS encoder");
156 MODULE_LICENSE("GPL");