|
@@ -16,14 +16,13 @@
|
|
|
#include <drm/drmP.h>
|
|
|
#include <drm/drm_crtc.h>
|
|
|
#include <drm/drm_crtc_helper.h>
|
|
|
+#include <drm/drm_panel.h>
|
|
|
|
|
|
#include "rcar_du_drv.h"
|
|
|
#include "rcar_du_encoder.h"
|
|
|
-#include "rcar_du_hdmienc.h"
|
|
|
#include "rcar_du_kms.h"
|
|
|
#include "rcar_du_lvdscon.h"
|
|
|
#include "rcar_du_lvdsenc.h"
|
|
|
-#include "rcar_du_vgacon.h"
|
|
|
|
|
|
/* -----------------------------------------------------------------------------
|
|
|
* Encoder
|
|
@@ -33,6 +32,11 @@ static void rcar_du_encoder_disable(struct drm_encoder *encoder)
|
|
|
{
|
|
|
struct rcar_du_encoder *renc = to_rcar_encoder(encoder);
|
|
|
|
|
|
+ if (renc->connector && renc->connector->panel) {
|
|
|
+ drm_panel_disable(renc->connector->panel);
|
|
|
+ drm_panel_unprepare(renc->connector->panel);
|
|
|
+ }
|
|
|
+
|
|
|
if (renc->lvds)
|
|
|
rcar_du_lvdsenc_enable(renc->lvds, encoder->crtc, false);
|
|
|
}
|
|
@@ -43,6 +47,11 @@ static void rcar_du_encoder_enable(struct drm_encoder *encoder)
|
|
|
|
|
|
if (renc->lvds)
|
|
|
rcar_du_lvdsenc_enable(renc->lvds, encoder->crtc, true);
|
|
|
+
|
|
|
+ if (renc->connector && renc->connector->panel) {
|
|
|
+ drm_panel_prepare(renc->connector->panel);
|
|
|
+ drm_panel_enable(renc->connector->panel);
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
static int rcar_du_encoder_atomic_check(struct drm_encoder *encoder,
|
|
@@ -52,30 +61,36 @@ static int rcar_du_encoder_atomic_check(struct drm_encoder *encoder,
|
|
|
struct rcar_du_encoder *renc = to_rcar_encoder(encoder);
|
|
|
struct drm_display_mode *adjusted_mode = &crtc_state->adjusted_mode;
|
|
|
const struct drm_display_mode *mode = &crtc_state->mode;
|
|
|
- const struct drm_display_mode *panel_mode;
|
|
|
struct drm_connector *connector = conn_state->connector;
|
|
|
struct drm_device *dev = encoder->dev;
|
|
|
|
|
|
- /* DAC encoders have currently no restriction on the mode. */
|
|
|
- if (encoder->encoder_type == DRM_MODE_ENCODER_DAC)
|
|
|
- return 0;
|
|
|
-
|
|
|
- if (list_empty(&connector->modes)) {
|
|
|
- dev_dbg(dev->dev, "encoder: empty modes list\n");
|
|
|
- return -EINVAL;
|
|
|
+ /*
|
|
|
+ * Only panel-related encoder types require validation here, everything
|
|
|
+ * else is handled by the bridge drivers.
|
|
|
+ */
|
|
|
+ if (connector->connector_type == DRM_MODE_CONNECTOR_LVDS) {
|
|
|
+ const struct drm_display_mode *panel_mode;
|
|
|
+
|
|
|
+ if (list_empty(&connector->modes)) {
|
|
|
+ dev_dbg(dev->dev, "encoder: empty modes list\n");
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ panel_mode = list_first_entry(&connector->modes,
|
|
|
+ struct drm_display_mode, head);
|
|
|
+
|
|
|
+ /* We're not allowed to modify the resolution. */
|
|
|
+ if (mode->hdisplay != panel_mode->hdisplay ||
|
|
|
+ mode->vdisplay != panel_mode->vdisplay)
|
|
|
+ return -EINVAL;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * The flat panel mode is fixed, just copy it to the adjusted
|
|
|
+ * mode.
|
|
|
+ */
|
|
|
+ drm_mode_copy(adjusted_mode, panel_mode);
|
|
|
}
|
|
|
|
|
|
- panel_mode = list_first_entry(&connector->modes,
|
|
|
- struct drm_display_mode, head);
|
|
|
-
|
|
|
- /* We're not allowed to modify the resolution. */
|
|
|
- if (mode->hdisplay != panel_mode->hdisplay ||
|
|
|
- mode->vdisplay != panel_mode->vdisplay)
|
|
|
- return -EINVAL;
|
|
|
-
|
|
|
- /* The flat panel mode is fixed, just copy it to the adjusted mode. */
|
|
|
- drm_mode_copy(adjusted_mode, panel_mode);
|
|
|
-
|
|
|
if (renc->lvds)
|
|
|
rcar_du_lvdsenc_atomic_check(renc->lvds, adjusted_mode);
|
|
|
|
|
@@ -83,16 +98,54 @@ static int rcar_du_encoder_atomic_check(struct drm_encoder *encoder,
|
|
|
}
|
|
|
|
|
|
static void rcar_du_encoder_mode_set(struct drm_encoder *encoder,
|
|
|
- struct drm_display_mode *mode,
|
|
|
- struct drm_display_mode *adjusted_mode)
|
|
|
+ struct drm_crtc_state *crtc_state,
|
|
|
+ struct drm_connector_state *conn_state)
|
|
|
{
|
|
|
struct rcar_du_encoder *renc = to_rcar_encoder(encoder);
|
|
|
+ struct drm_display_info *info = &conn_state->connector->display_info;
|
|
|
+ enum rcar_lvds_mode mode;
|
|
|
+
|
|
|
+ rcar_du_crtc_route_output(crtc_state->crtc, renc->output);
|
|
|
+
|
|
|
+ if (!renc->lvds) {
|
|
|
+ /*
|
|
|
+ * The DU driver creates connectors only for the outputs of the
|
|
|
+ * internal LVDS encoders.
|
|
|
+ */
|
|
|
+ renc->connector = NULL;
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ renc->connector = to_rcar_connector(conn_state->connector);
|
|
|
+
|
|
|
+ if (!info->num_bus_formats || !info->bus_formats) {
|
|
|
+ dev_err(encoder->dev->dev, "no LVDS bus format reported\n");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ switch (info->bus_formats[0]) {
|
|
|
+ case MEDIA_BUS_FMT_RGB666_1X7X3_SPWG:
|
|
|
+ case MEDIA_BUS_FMT_RGB888_1X7X4_JEIDA:
|
|
|
+ mode = RCAR_LVDS_MODE_JEIDA;
|
|
|
+ break;
|
|
|
+ case MEDIA_BUS_FMT_RGB888_1X7X4_SPWG:
|
|
|
+ mode = RCAR_LVDS_MODE_VESA;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ dev_err(encoder->dev->dev,
|
|
|
+ "unsupported LVDS bus format 0x%04x\n",
|
|
|
+ info->bus_formats[0]);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (info->bus_flags & DRM_BUS_FLAG_DATA_LSB_TO_MSB)
|
|
|
+ mode |= RCAR_LVDS_MODE_MIRROR;
|
|
|
|
|
|
- rcar_du_crtc_route_output(encoder->crtc, renc->output);
|
|
|
+ rcar_du_lvdsenc_set_mode(renc->lvds, mode);
|
|
|
}
|
|
|
|
|
|
static const struct drm_encoder_helper_funcs encoder_helper_funcs = {
|
|
|
- .mode_set = rcar_du_encoder_mode_set,
|
|
|
+ .atomic_mode_set = rcar_du_encoder_mode_set,
|
|
|
.disable = rcar_du_encoder_disable,
|
|
|
.enable = rcar_du_encoder_enable,
|
|
|
.atomic_check = rcar_du_encoder_atomic_check,
|
|
@@ -103,14 +156,13 @@ static const struct drm_encoder_funcs encoder_funcs = {
|
|
|
};
|
|
|
|
|
|
int rcar_du_encoder_init(struct rcar_du_device *rcdu,
|
|
|
- enum rcar_du_encoder_type type,
|
|
|
enum rcar_du_output output,
|
|
|
struct device_node *enc_node,
|
|
|
struct device_node *con_node)
|
|
|
{
|
|
|
struct rcar_du_encoder *renc;
|
|
|
struct drm_encoder *encoder;
|
|
|
- unsigned int encoder_type;
|
|
|
+ struct drm_bridge *bridge = NULL;
|
|
|
int ret;
|
|
|
|
|
|
renc = devm_kzalloc(rcdu->dev, sizeof(*renc), GFP_KERNEL);
|
|
@@ -133,52 +185,51 @@ int rcar_du_encoder_init(struct rcar_du_device *rcdu,
|
|
|
break;
|
|
|
}
|
|
|
|
|
|
- switch (type) {
|
|
|
- case RCAR_DU_ENCODER_VGA:
|
|
|
- encoder_type = DRM_MODE_ENCODER_DAC;
|
|
|
- break;
|
|
|
- case RCAR_DU_ENCODER_LVDS:
|
|
|
- encoder_type = DRM_MODE_ENCODER_LVDS;
|
|
|
- break;
|
|
|
- case RCAR_DU_ENCODER_HDMI:
|
|
|
- encoder_type = DRM_MODE_ENCODER_TMDS;
|
|
|
- break;
|
|
|
- case RCAR_DU_ENCODER_NONE:
|
|
|
- default:
|
|
|
- /* No external encoder, use the internal encoder type. */
|
|
|
- encoder_type = rcdu->info->routes[output].encoder_type;
|
|
|
- break;
|
|
|
- }
|
|
|
+ if (enc_node) {
|
|
|
+ dev_dbg(rcdu->dev, "initializing encoder %s for output %u\n",
|
|
|
+ of_node_full_name(enc_node), output);
|
|
|
|
|
|
- if (type == RCAR_DU_ENCODER_HDMI) {
|
|
|
- ret = rcar_du_hdmienc_init(rcdu, renc, enc_node);
|
|
|
- if (ret < 0)
|
|
|
+ /* Locate the DRM bridge from the encoder DT node. */
|
|
|
+ bridge = of_drm_find_bridge(enc_node);
|
|
|
+ if (!bridge) {
|
|
|
+ ret = -EPROBE_DEFER;
|
|
|
goto done;
|
|
|
+ }
|
|
|
} else {
|
|
|
- ret = drm_encoder_init(rcdu->ddev, encoder, &encoder_funcs,
|
|
|
- encoder_type, NULL);
|
|
|
- if (ret < 0)
|
|
|
- goto done;
|
|
|
-
|
|
|
- drm_encoder_helper_add(encoder, &encoder_helper_funcs);
|
|
|
+ dev_dbg(rcdu->dev,
|
|
|
+ "initializing internal encoder for output %u\n",
|
|
|
+ output);
|
|
|
}
|
|
|
|
|
|
- switch (encoder_type) {
|
|
|
- case DRM_MODE_ENCODER_LVDS:
|
|
|
- ret = rcar_du_lvds_connector_init(rcdu, renc, con_node);
|
|
|
- break;
|
|
|
-
|
|
|
- case DRM_MODE_ENCODER_DAC:
|
|
|
- ret = rcar_du_vga_connector_init(rcdu, renc);
|
|
|
- break;
|
|
|
-
|
|
|
- case DRM_MODE_ENCODER_TMDS:
|
|
|
- /* connector managed by the bridge driver */
|
|
|
- break;
|
|
|
-
|
|
|
- default:
|
|
|
- ret = -EINVAL;
|
|
|
- break;
|
|
|
+ ret = drm_encoder_init(rcdu->ddev, encoder, &encoder_funcs,
|
|
|
+ DRM_MODE_ENCODER_NONE, NULL);
|
|
|
+ if (ret < 0)
|
|
|
+ goto done;
|
|
|
+
|
|
|
+ drm_encoder_helper_add(encoder, &encoder_helper_funcs);
|
|
|
+
|
|
|
+ if (bridge) {
|
|
|
+ /*
|
|
|
+ * Attach the bridge to the encoder. The bridge will create the
|
|
|
+ * connector.
|
|
|
+ */
|
|
|
+ ret = drm_bridge_attach(encoder, bridge, NULL);
|
|
|
+ if (ret) {
|
|
|
+ drm_encoder_cleanup(encoder);
|
|
|
+ return ret;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ /* There's no bridge, create the connector manually. */
|
|
|
+ switch (output) {
|
|
|
+ case RCAR_DU_OUTPUT_LVDS0:
|
|
|
+ case RCAR_DU_OUTPUT_LVDS1:
|
|
|
+ ret = rcar_du_lvds_connector_init(rcdu, renc, con_node);
|
|
|
+ break;
|
|
|
+
|
|
|
+ default:
|
|
|
+ ret = -EINVAL;
|
|
|
+ break;
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
done:
|