mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-20 02:34:23 +08:00
drm/sun4i: Add bridge support
Our RGB bus can be either connected to a bridge or a panel. While the panel support was already there, the bridge was not. Fix that. Signed-off-by: Maxime Ripard <maxime.ripard@free-electrons.com>
This commit is contained in:
parent
a8444c7ee2
commit
894f5a9f4b
@ -257,8 +257,8 @@ static int sun4i_drv_add_endpoints(struct device *dev,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the node is our TCON, the first port is used for our
|
* If the node is our TCON, the first port is used for
|
||||||
* panel, and will not be part of the
|
* panel or bridges, and will not be part of the
|
||||||
* component framework.
|
* component framework.
|
||||||
*/
|
*/
|
||||||
if (sun4i_drv_node_is_tcon(node)) {
|
if (sun4i_drv_node_is_tcon(node)) {
|
||||||
|
@ -151,7 +151,12 @@ static void sun4i_rgb_encoder_enable(struct drm_encoder *encoder)
|
|||||||
|
|
||||||
DRM_DEBUG_DRIVER("Enabling RGB output\n");
|
DRM_DEBUG_DRIVER("Enabling RGB output\n");
|
||||||
|
|
||||||
|
if (!IS_ERR(tcon->panel))
|
||||||
drm_panel_enable(tcon->panel);
|
drm_panel_enable(tcon->panel);
|
||||||
|
|
||||||
|
if (!IS_ERR(encoder->bridge))
|
||||||
|
drm_bridge_enable(encoder->bridge);
|
||||||
|
|
||||||
sun4i_tcon_channel_enable(tcon, 0);
|
sun4i_tcon_channel_enable(tcon, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -164,6 +169,11 @@ static void sun4i_rgb_encoder_disable(struct drm_encoder *encoder)
|
|||||||
DRM_DEBUG_DRIVER("Disabling RGB output\n");
|
DRM_DEBUG_DRIVER("Disabling RGB output\n");
|
||||||
|
|
||||||
sun4i_tcon_channel_disable(tcon, 0);
|
sun4i_tcon_channel_disable(tcon, 0);
|
||||||
|
|
||||||
|
if (!IS_ERR(encoder->bridge))
|
||||||
|
drm_bridge_disable(encoder->bridge);
|
||||||
|
|
||||||
|
if (!IS_ERR(tcon->panel))
|
||||||
drm_panel_disable(tcon->panel);
|
drm_panel_disable(tcon->panel);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -203,6 +213,7 @@ int sun4i_rgb_init(struct drm_device *drm)
|
|||||||
{
|
{
|
||||||
struct sun4i_drv *drv = drm->dev_private;
|
struct sun4i_drv *drv = drm->dev_private;
|
||||||
struct sun4i_tcon *tcon = drv->tcon;
|
struct sun4i_tcon *tcon = drv->tcon;
|
||||||
|
struct drm_encoder *encoder;
|
||||||
struct sun4i_rgb *rgb;
|
struct sun4i_rgb *rgb;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
@ -210,10 +221,12 @@ int sun4i_rgb_init(struct drm_device *drm)
|
|||||||
if (!rgb)
|
if (!rgb)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
rgb->drv = drv;
|
rgb->drv = drv;
|
||||||
|
encoder = &rgb->encoder;
|
||||||
|
|
||||||
tcon->panel = sun4i_tcon_find_panel(tcon->dev->of_node);
|
tcon->panel = sun4i_tcon_find_panel(tcon->dev->of_node);
|
||||||
if (IS_ERR(tcon->panel)) {
|
encoder->bridge = sun4i_tcon_find_bridge(tcon->dev->of_node);
|
||||||
dev_info(drm->dev, "No panel found... RGB output disabled\n");
|
if (IS_ERR(tcon->panel) && IS_ERR(encoder->bridge)) {
|
||||||
|
dev_info(drm->dev, "No panel or bridge found... RGB output disabled\n");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -232,6 +245,7 @@ int sun4i_rgb_init(struct drm_device *drm)
|
|||||||
/* The RGB encoder can only work with the TCON channel 0 */
|
/* The RGB encoder can only work with the TCON channel 0 */
|
||||||
rgb->encoder.possible_crtcs = BIT(0);
|
rgb->encoder.possible_crtcs = BIT(0);
|
||||||
|
|
||||||
|
if (!IS_ERR(tcon->panel)) {
|
||||||
drm_connector_helper_add(&rgb->connector,
|
drm_connector_helper_add(&rgb->connector,
|
||||||
&sun4i_rgb_con_helper_funcs);
|
&sun4i_rgb_con_helper_funcs);
|
||||||
ret = drm_connector_init(drm, &rgb->connector,
|
ret = drm_connector_init(drm, &rgb->connector,
|
||||||
@ -242,9 +256,25 @@ int sun4i_rgb_init(struct drm_device *drm)
|
|||||||
goto err_cleanup_connector;
|
goto err_cleanup_connector;
|
||||||
}
|
}
|
||||||
|
|
||||||
drm_mode_connector_attach_encoder(&rgb->connector, &rgb->encoder);
|
drm_mode_connector_attach_encoder(&rgb->connector,
|
||||||
|
&rgb->encoder);
|
||||||
|
|
||||||
drm_panel_attach(tcon->panel, &rgb->connector);
|
ret = drm_panel_attach(tcon->panel, &rgb->connector);
|
||||||
|
if (ret) {
|
||||||
|
dev_err(drm->dev, "Couldn't attach our panel\n");
|
||||||
|
goto err_cleanup_connector;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!IS_ERR(encoder->bridge)) {
|
||||||
|
encoder->bridge->encoder = &rgb->encoder;
|
||||||
|
|
||||||
|
ret = drm_bridge_attach(drm, encoder->bridge);
|
||||||
|
if (ret) {
|
||||||
|
dev_err(drm->dev, "Couldn't attach our bridge\n");
|
||||||
|
goto err_cleanup_connector;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
@ -432,6 +432,40 @@ struct drm_panel *sun4i_tcon_find_panel(struct device_node *node)
|
|||||||
return of_drm_find_panel(remote) ?: ERR_PTR(-EPROBE_DEFER);
|
return of_drm_find_panel(remote) ?: ERR_PTR(-EPROBE_DEFER);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct drm_bridge *sun4i_tcon_find_bridge(struct device_node *node)
|
||||||
|
{
|
||||||
|
struct device_node *port, *remote, *child;
|
||||||
|
struct device_node *end_node = NULL;
|
||||||
|
|
||||||
|
/* Inputs are listed first, then outputs */
|
||||||
|
port = of_graph_get_port_by_id(node, 1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Our first output is the RGB interface where the panel will
|
||||||
|
* be connected.
|
||||||
|
*/
|
||||||
|
for_each_child_of_node(port, child) {
|
||||||
|
u32 reg;
|
||||||
|
|
||||||
|
of_property_read_u32(child, "reg", ®);
|
||||||
|
if (reg == 0)
|
||||||
|
end_node = child;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!end_node) {
|
||||||
|
DRM_DEBUG_DRIVER("Missing bridge endpoint\n");
|
||||||
|
return ERR_PTR(-ENODEV);
|
||||||
|
}
|
||||||
|
|
||||||
|
remote = of_graph_get_remote_port_parent(end_node);
|
||||||
|
if (!remote) {
|
||||||
|
DRM_DEBUG_DRIVER("Enable to parse remote node\n");
|
||||||
|
return ERR_PTR(-EINVAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
return of_drm_find_bridge(remote) ?: ERR_PTR(-EPROBE_DEFER);
|
||||||
|
}
|
||||||
|
|
||||||
static int sun4i_tcon_bind(struct device *dev, struct device *master,
|
static int sun4i_tcon_bind(struct device *dev, struct device *master,
|
||||||
void *data)
|
void *data)
|
||||||
{
|
{
|
||||||
@ -514,19 +548,22 @@ static struct component_ops sun4i_tcon_ops = {
|
|||||||
static int sun4i_tcon_probe(struct platform_device *pdev)
|
static int sun4i_tcon_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct device_node *node = pdev->dev.of_node;
|
struct device_node *node = pdev->dev.of_node;
|
||||||
|
struct drm_bridge *bridge;
|
||||||
struct drm_panel *panel;
|
struct drm_panel *panel;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The panel is not ready.
|
* Neither the bridge or the panel is ready.
|
||||||
* Defer the probe.
|
* Defer the probe.
|
||||||
*/
|
*/
|
||||||
panel = sun4i_tcon_find_panel(node);
|
panel = sun4i_tcon_find_panel(node);
|
||||||
|
bridge = sun4i_tcon_find_bridge(node);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If we don't have a panel endpoint, just go on
|
* If we don't have a panel endpoint, just go on
|
||||||
*/
|
*/
|
||||||
if (PTR_ERR(panel) == -EPROBE_DEFER) {
|
if ((PTR_ERR(panel) == -EPROBE_DEFER) &&
|
||||||
DRM_DEBUG_DRIVER("Still waiting for our panel. Deferring...\n");
|
(PTR_ERR(bridge) == -EPROBE_DEFER)) {
|
||||||
|
DRM_DEBUG_DRIVER("Still waiting for our panel/bridge. Deferring...\n");
|
||||||
return -EPROBE_DEFER;
|
return -EPROBE_DEFER;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -166,6 +166,7 @@ struct sun4i_tcon {
|
|||||||
struct drm_panel *panel;
|
struct drm_panel *panel;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct drm_bridge *sun4i_tcon_find_bridge(struct device_node *node);
|
||||||
struct drm_panel *sun4i_tcon_find_panel(struct device_node *node);
|
struct drm_panel *sun4i_tcon_find_panel(struct device_node *node);
|
||||||
|
|
||||||
/* Global Control */
|
/* Global Control */
|
||||||
|
Loading…
Reference in New Issue
Block a user