drm/imx: convert imx-drm to use the generic DRM OF helper
Use the generic DRM OF helper to locate the possible CRTCs for the encoder, thereby shrinking the imx-drm driver some more. Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk> Signed-off-by: Philipp Zabel <p.zabel@pengutronix.de>
This commit is contained in:
parent
f4876ffea6
commit
6457b9716b
|
@ -25,6 +25,7 @@
|
||||||
#include <drm/drm_gem_cma_helper.h>
|
#include <drm/drm_gem_cma_helper.h>
|
||||||
#include <drm/drm_fb_cma_helper.h>
|
#include <drm/drm_fb_cma_helper.h>
|
||||||
#include <drm/drm_plane_helper.h>
|
#include <drm/drm_plane_helper.h>
|
||||||
|
#include <drm/drm_of.h>
|
||||||
|
|
||||||
#include "imx-drm.h"
|
#include "imx-drm.h"
|
||||||
|
|
||||||
|
@ -46,7 +47,6 @@ struct imx_drm_crtc {
|
||||||
struct drm_crtc *crtc;
|
struct drm_crtc *crtc;
|
||||||
int pipe;
|
int pipe;
|
||||||
struct imx_drm_crtc_helper_funcs imx_drm_helper_funcs;
|
struct imx_drm_crtc_helper_funcs imx_drm_helper_funcs;
|
||||||
struct device_node *port;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static int legacyfb_depth = 16;
|
static int legacyfb_depth = 16;
|
||||||
|
@ -365,9 +365,10 @@ int imx_drm_add_crtc(struct drm_device *drm, struct drm_crtc *crtc,
|
||||||
|
|
||||||
imx_drm_crtc->imx_drm_helper_funcs = *imx_drm_helper_funcs;
|
imx_drm_crtc->imx_drm_helper_funcs = *imx_drm_helper_funcs;
|
||||||
imx_drm_crtc->pipe = imxdrm->pipes++;
|
imx_drm_crtc->pipe = imxdrm->pipes++;
|
||||||
imx_drm_crtc->port = port;
|
|
||||||
imx_drm_crtc->crtc = crtc;
|
imx_drm_crtc->crtc = crtc;
|
||||||
|
|
||||||
|
crtc->port = port;
|
||||||
|
|
||||||
imxdrm->crtc[imx_drm_crtc->pipe] = imx_drm_crtc;
|
imxdrm->crtc[imx_drm_crtc->pipe] = imx_drm_crtc;
|
||||||
|
|
||||||
*new_crtc = imx_drm_crtc;
|
*new_crtc = imx_drm_crtc;
|
||||||
|
@ -408,75 +409,19 @@ int imx_drm_remove_crtc(struct imx_drm_crtc *imx_drm_crtc)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(imx_drm_remove_crtc);
|
EXPORT_SYMBOL_GPL(imx_drm_remove_crtc);
|
||||||
|
|
||||||
/*
|
|
||||||
* Find the DRM CRTC possible mask for the connected endpoint.
|
|
||||||
*
|
|
||||||
* The encoder possible masks are defined by their position in the
|
|
||||||
* mode_config crtc_list. This means that CRTCs must not be added
|
|
||||||
* or removed once the DRM device has been fully initialised.
|
|
||||||
*/
|
|
||||||
static uint32_t imx_drm_find_crtc_mask(struct imx_drm_device *imxdrm,
|
|
||||||
struct device_node *endpoint)
|
|
||||||
{
|
|
||||||
struct device_node *port;
|
|
||||||
unsigned i;
|
|
||||||
|
|
||||||
port = of_graph_get_remote_port(endpoint);
|
|
||||||
if (!port)
|
|
||||||
return 0;
|
|
||||||
of_node_put(port);
|
|
||||||
|
|
||||||
for (i = 0; i < MAX_CRTC; i++) {
|
|
||||||
struct imx_drm_crtc *imx_drm_crtc = imxdrm->crtc[i];
|
|
||||||
|
|
||||||
if (imx_drm_crtc && imx_drm_crtc->port == port)
|
|
||||||
return drm_crtc_mask(imx_drm_crtc->crtc);
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct device_node *imx_drm_of_get_next_endpoint(
|
|
||||||
const struct device_node *parent, struct device_node *prev)
|
|
||||||
{
|
|
||||||
struct device_node *node = of_graph_get_next_endpoint(parent, prev);
|
|
||||||
|
|
||||||
of_node_put(prev);
|
|
||||||
return node;
|
|
||||||
}
|
|
||||||
|
|
||||||
int imx_drm_encoder_parse_of(struct drm_device *drm,
|
int imx_drm_encoder_parse_of(struct drm_device *drm,
|
||||||
struct drm_encoder *encoder, struct device_node *np)
|
struct drm_encoder *encoder, struct device_node *np)
|
||||||
{
|
{
|
||||||
struct imx_drm_device *imxdrm = drm->dev_private;
|
uint32_t crtc_mask = drm_of_find_possible_crtcs(drm, np);
|
||||||
struct device_node *ep = NULL;
|
|
||||||
uint32_t crtc_mask = 0;
|
|
||||||
int i;
|
|
||||||
|
|
||||||
for (i = 0; ; i++) {
|
/*
|
||||||
u32 mask;
|
* If we failed to find the CRTC(s) which this encoder is
|
||||||
|
* supposed to be connected to, it's because the CRTC has
|
||||||
ep = imx_drm_of_get_next_endpoint(np, ep);
|
* not been registered yet. Defer probing, and hope that
|
||||||
if (!ep)
|
* the required CRTC is added later.
|
||||||
break;
|
*/
|
||||||
|
if (crtc_mask == 0)
|
||||||
mask = imx_drm_find_crtc_mask(imxdrm, ep);
|
return -EPROBE_DEFER;
|
||||||
|
|
||||||
/*
|
|
||||||
* If we failed to find the CRTC(s) which this encoder is
|
|
||||||
* supposed to be connected to, it's because the CRTC has
|
|
||||||
* not been registered yet. Defer probing, and hope that
|
|
||||||
* the required CRTC is added later.
|
|
||||||
*/
|
|
||||||
if (mask == 0)
|
|
||||||
return -EPROBE_DEFER;
|
|
||||||
|
|
||||||
crtc_mask |= mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
of_node_put(ep);
|
|
||||||
if (i == 0)
|
|
||||||
return -ENOENT;
|
|
||||||
|
|
||||||
encoder->possible_crtcs = crtc_mask;
|
encoder->possible_crtcs = crtc_mask;
|
||||||
|
|
||||||
|
@ -487,6 +432,15 @@ int imx_drm_encoder_parse_of(struct drm_device *drm,
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(imx_drm_encoder_parse_of);
|
EXPORT_SYMBOL_GPL(imx_drm_encoder_parse_of);
|
||||||
|
|
||||||
|
static struct device_node *imx_drm_of_get_next_endpoint(
|
||||||
|
const struct device_node *parent, struct device_node *prev)
|
||||||
|
{
|
||||||
|
struct device_node *node = of_graph_get_next_endpoint(parent, prev);
|
||||||
|
|
||||||
|
of_node_put(prev);
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @node: device tree node containing encoder input ports
|
* @node: device tree node containing encoder input ports
|
||||||
* @encoder: drm_encoder
|
* @encoder: drm_encoder
|
||||||
|
@ -510,7 +464,7 @@ int imx_drm_encoder_get_mux_id(struct device_node *node,
|
||||||
|
|
||||||
port = of_graph_get_remote_port(ep);
|
port = of_graph_get_remote_port(ep);
|
||||||
of_node_put(port);
|
of_node_put(port);
|
||||||
if (port == imx_crtc->port) {
|
if (port == imx_crtc->crtc->port) {
|
||||||
ret = of_graph_parse_endpoint(ep, &endpoint);
|
ret = of_graph_parse_endpoint(ep, &endpoint);
|
||||||
return ret ? ret : endpoint.port;
|
return ret ? ret : endpoint.port;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue