mirror of
https://github.com/hardkernel/linux.git
synced 2026-06-07 03:15:31 +09:00
Merge tag 'du-next-20211007' of git://linuxtv.org/pinchartl/media into drm-next
R-Car DU improvements & enhancements to misc drivers Most notably, - Non-contiguous buffer import support for rcar-du - r8a779a0 support preparation for rcar-du - COMPILE_TEST fixes for omapdrm and sti Signed-off-by: Dave Airlie <airlied@redhat.com> From: Laurent Pinchart <laurent.pinchart@ideasonboard.com> Link: https://patchwork.freedesktop.org/patch/msgid/YV5jfi+/qjTJKeb3@pendragon.ideasonboard.com
This commit is contained in:
@@ -39,6 +39,7 @@ properties:
|
|||||||
- renesas,du-r8a77980 # for R-Car V3H compatible DU
|
- renesas,du-r8a77980 # for R-Car V3H compatible DU
|
||||||
- renesas,du-r8a77990 # for R-Car E3 compatible DU
|
- renesas,du-r8a77990 # for R-Car E3 compatible DU
|
||||||
- renesas,du-r8a77995 # for R-Car D3 compatible DU
|
- renesas,du-r8a77995 # for R-Car D3 compatible DU
|
||||||
|
- renesas,du-r8a779a0 # for R-Car V3U compatible DU
|
||||||
|
|
||||||
reg:
|
reg:
|
||||||
maxItems: 1
|
maxItems: 1
|
||||||
@@ -773,6 +774,56 @@ allOf:
|
|||||||
- reset-names
|
- reset-names
|
||||||
- renesas,vsps
|
- renesas,vsps
|
||||||
|
|
||||||
|
- if:
|
||||||
|
properties:
|
||||||
|
compatible:
|
||||||
|
contains:
|
||||||
|
enum:
|
||||||
|
- renesas,du-r8a779a0
|
||||||
|
then:
|
||||||
|
properties:
|
||||||
|
clocks:
|
||||||
|
items:
|
||||||
|
- description: Functional clock
|
||||||
|
|
||||||
|
clock-names:
|
||||||
|
maxItems: 1
|
||||||
|
items:
|
||||||
|
- const: du.0
|
||||||
|
|
||||||
|
interrupts:
|
||||||
|
maxItems: 2
|
||||||
|
|
||||||
|
resets:
|
||||||
|
maxItems: 1
|
||||||
|
|
||||||
|
reset-names:
|
||||||
|
items:
|
||||||
|
- const: du.0
|
||||||
|
|
||||||
|
ports:
|
||||||
|
properties:
|
||||||
|
port@0:
|
||||||
|
description: DSI 0
|
||||||
|
port@1:
|
||||||
|
description: DSI 1
|
||||||
|
port@2: false
|
||||||
|
port@3: false
|
||||||
|
|
||||||
|
required:
|
||||||
|
- port@0
|
||||||
|
- port@1
|
||||||
|
|
||||||
|
renesas,vsps:
|
||||||
|
minItems: 2
|
||||||
|
|
||||||
|
required:
|
||||||
|
- clock-names
|
||||||
|
- interrupts
|
||||||
|
- resets
|
||||||
|
- reset-names
|
||||||
|
- renesas,vsps
|
||||||
|
|
||||||
additionalProperties: false
|
additionalProperties: false
|
||||||
|
|
||||||
examples:
|
examples:
|
||||||
|
|||||||
@@ -127,8 +127,7 @@ struct drm_property *drm_property_create(struct drm_device *dev,
|
|||||||
property->num_values = num_values;
|
property->num_values = num_values;
|
||||||
INIT_LIST_HEAD(&property->enum_list);
|
INIT_LIST_HEAD(&property->enum_list);
|
||||||
|
|
||||||
strncpy(property->name, name, DRM_PROP_NAME_LEN);
|
strscpy_pad(property->name, name, DRM_PROP_NAME_LEN);
|
||||||
property->name[DRM_PROP_NAME_LEN-1] = '\0';
|
|
||||||
|
|
||||||
list_add_tail(&property->head, &dev->mode_config.property_list);
|
list_add_tail(&property->head, &dev->mode_config.property_list);
|
||||||
|
|
||||||
@@ -421,8 +420,7 @@ int drm_property_add_enum(struct drm_property *property,
|
|||||||
if (!prop_enum)
|
if (!prop_enum)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
strncpy(prop_enum->name, name, DRM_PROP_NAME_LEN);
|
strscpy_pad(prop_enum->name, name, DRM_PROP_NAME_LEN);
|
||||||
prop_enum->name[DRM_PROP_NAME_LEN-1] = '\0';
|
|
||||||
prop_enum->value = value;
|
prop_enum->value = value;
|
||||||
|
|
||||||
property->values[index] = value;
|
property->values[index] = value;
|
||||||
@@ -475,8 +473,7 @@ int drm_mode_getproperty_ioctl(struct drm_device *dev,
|
|||||||
if (!property)
|
if (!property)
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
|
|
||||||
strncpy(out_resp->name, property->name, DRM_PROP_NAME_LEN);
|
strscpy_pad(out_resp->name, property->name, DRM_PROP_NAME_LEN);
|
||||||
out_resp->name[DRM_PROP_NAME_LEN-1] = 0;
|
|
||||||
out_resp->flags = property->flags;
|
out_resp->flags = property->flags;
|
||||||
|
|
||||||
value_count = property->num_values;
|
value_count = property->num_values;
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
# SPDX-License-Identifier: GPL-2.0-only
|
# SPDX-License-Identifier: GPL-2.0-only
|
||||||
config DRM_OMAP
|
config DRM_OMAP
|
||||||
tristate "OMAP DRM"
|
tristate "OMAP DRM"
|
||||||
depends on DRM
|
depends on DRM && OF
|
||||||
depends on ARCH_OMAP2PLUS || ARCH_MULTIPLATFORM
|
depends on ARCH_OMAP2PLUS || ARCH_MULTIPLATFORM
|
||||||
select DRM_KMS_HELPER
|
select DRM_KMS_HELPER
|
||||||
select VIDEOMODE_HELPERS
|
select VIDEOMODE_HELPERS
|
||||||
|
|||||||
@@ -2094,7 +2094,7 @@ static int dsi_vc_send_long(struct dsi_data *dsi, int vc,
|
|||||||
u8 b1, b2, b3, b4;
|
u8 b1, b2, b3, b4;
|
||||||
|
|
||||||
if (dsi->debug_write)
|
if (dsi->debug_write)
|
||||||
DSSDBG("dsi_vc_send_long, %d bytes\n", msg->tx_len);
|
DSSDBG("dsi_vc_send_long, %zu bytes\n", msg->tx_len);
|
||||||
|
|
||||||
/* len + header */
|
/* len + header */
|
||||||
if (dsi->vc[vc].tx_fifo_size * 32 * 4 < msg->tx_len + 4) {
|
if (dsi->vc[vc].tx_fifo_size * 32 * 4 < msg->tx_len + 4) {
|
||||||
@@ -2390,7 +2390,7 @@ static int dsi_vc_generic_read(struct omap_dss_device *dssdev, int vc,
|
|||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
err:
|
err:
|
||||||
DSSERR("%s(vc %d, reqlen %d) failed\n", __func__, vc, msg->tx_len);
|
DSSERR("%s(vc %d, reqlen %zu) failed\n", __func__, vc, msg->tx_len);
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -572,7 +572,7 @@ static int omapdrm_init(struct omap_drm_private *priv, struct device *dev)
|
|||||||
priv->dss->mgr_ops_priv = priv;
|
priv->dss->mgr_ops_priv = priv;
|
||||||
|
|
||||||
soc = soc_device_match(omapdrm_soc_devices);
|
soc = soc_device_match(omapdrm_soc_devices);
|
||||||
priv->omaprev = soc ? (unsigned int)soc->data : 0;
|
priv->omaprev = soc ? (uintptr_t)soc->data : 0;
|
||||||
priv->wq = alloc_ordered_workqueue("omapdrm", 0);
|
priv->wq = alloc_ordered_workqueue("omapdrm", 0);
|
||||||
|
|
||||||
mutex_init(&priv->list_lock);
|
mutex_init(&priv->list_lock);
|
||||||
|
|||||||
@@ -1206,7 +1206,7 @@ int rcar_du_crtc_create(struct rcar_du_group *rgrp, unsigned int swindex,
|
|||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
/* Get the CRTC clock and the optional external clock. */
|
/* Get the CRTC clock and the optional external clock. */
|
||||||
if (rcar_du_has(rcdu, RCAR_DU_FEATURE_CRTC_IRQ_CLOCK)) {
|
if (rcar_du_has(rcdu, RCAR_DU_FEATURE_CRTC_CLOCK)) {
|
||||||
sprintf(clk_name, "du.%u", hwindex);
|
sprintf(clk_name, "du.%u", hwindex);
|
||||||
name = clk_name;
|
name = clk_name;
|
||||||
} else {
|
} else {
|
||||||
@@ -1243,7 +1243,10 @@ int rcar_du_crtc_create(struct rcar_du_group *rgrp, unsigned int swindex,
|
|||||||
rcrtc->group = rgrp;
|
rcrtc->group = rgrp;
|
||||||
rcrtc->mmio_offset = mmio_offsets[hwindex];
|
rcrtc->mmio_offset = mmio_offsets[hwindex];
|
||||||
rcrtc->index = hwindex;
|
rcrtc->index = hwindex;
|
||||||
rcrtc->dsysr = (rcrtc->index % 2 ? 0 : DSYSR_DRES) | DSYSR_TVM_TVSYNC;
|
rcrtc->dsysr = rcrtc->index % 2 ? 0 : DSYSR_DRES;
|
||||||
|
|
||||||
|
if (rcar_du_has(rcdu, RCAR_DU_FEATURE_TVM_SYNC))
|
||||||
|
rcrtc->dsysr |= DSYSR_TVM_TVSYNC;
|
||||||
|
|
||||||
if (rcar_du_has(rcdu, RCAR_DU_FEATURE_VSP1_SOURCE))
|
if (rcar_du_has(rcdu, RCAR_DU_FEATURE_VSP1_SOURCE))
|
||||||
primary = &rcrtc->vsp->planes[rcrtc->vsp_pipe].plane;
|
primary = &rcrtc->vsp->planes[rcrtc->vsp_pipe].plane;
|
||||||
@@ -1269,7 +1272,7 @@ int rcar_du_crtc_create(struct rcar_du_group *rgrp, unsigned int swindex,
|
|||||||
drm_crtc_helper_add(crtc, &crtc_helper_funcs);
|
drm_crtc_helper_add(crtc, &crtc_helper_funcs);
|
||||||
|
|
||||||
/* Register the interrupt handler. */
|
/* Register the interrupt handler. */
|
||||||
if (rcar_du_has(rcdu, RCAR_DU_FEATURE_CRTC_IRQ_CLOCK)) {
|
if (rcar_du_has(rcdu, RCAR_DU_FEATURE_CRTC_IRQ)) {
|
||||||
/* The IRQ's are associated with the CRTC (sw)index. */
|
/* The IRQ's are associated with the CRTC (sw)index. */
|
||||||
irq = platform_get_irq(pdev, swindex);
|
irq = platform_get_irq(pdev, swindex);
|
||||||
irqflags = 0;
|
irqflags = 0;
|
||||||
|
|||||||
@@ -93,17 +93,6 @@ struct rcar_du_crtc_state {
|
|||||||
|
|
||||||
#define to_rcar_crtc_state(s) container_of(s, struct rcar_du_crtc_state, state)
|
#define to_rcar_crtc_state(s) container_of(s, struct rcar_du_crtc_state, state)
|
||||||
|
|
||||||
enum rcar_du_output {
|
|
||||||
RCAR_DU_OUTPUT_DPAD0,
|
|
||||||
RCAR_DU_OUTPUT_DPAD1,
|
|
||||||
RCAR_DU_OUTPUT_LVDS0,
|
|
||||||
RCAR_DU_OUTPUT_LVDS1,
|
|
||||||
RCAR_DU_OUTPUT_HDMI0,
|
|
||||||
RCAR_DU_OUTPUT_HDMI1,
|
|
||||||
RCAR_DU_OUTPUT_TCON,
|
|
||||||
RCAR_DU_OUTPUT_MAX,
|
|
||||||
};
|
|
||||||
|
|
||||||
int rcar_du_crtc_create(struct rcar_du_group *rgrp, unsigned int swindex,
|
int rcar_du_crtc_create(struct rcar_du_group *rgrp, unsigned int swindex,
|
||||||
unsigned int hwindex);
|
unsigned int hwindex);
|
||||||
|
|
||||||
|
|||||||
@@ -8,6 +8,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/clk.h>
|
#include <linux/clk.h>
|
||||||
|
#include <linux/dma-mapping.h>
|
||||||
#include <linux/io.h>
|
#include <linux/io.h>
|
||||||
#include <linux/mm.h>
|
#include <linux/mm.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
@@ -36,7 +37,8 @@
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rzg1_du_r8a7743_info = {
|
static const struct rcar_du_device_info rzg1_du_r8a7743_info = {
|
||||||
.gen = 2,
|
.gen = 2,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
.channels_mask = BIT(1) | BIT(0),
|
.channels_mask = BIT(1) | BIT(0),
|
||||||
@@ -58,7 +60,8 @@ static const struct rcar_du_device_info rzg1_du_r8a7743_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rzg1_du_r8a7745_info = {
|
static const struct rcar_du_device_info rzg1_du_r8a7745_info = {
|
||||||
.gen = 2,
|
.gen = 2,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
.channels_mask = BIT(1) | BIT(0),
|
.channels_mask = BIT(1) | BIT(0),
|
||||||
@@ -79,7 +82,8 @@ static const struct rcar_du_device_info rzg1_du_r8a7745_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rzg1_du_r8a77470_info = {
|
static const struct rcar_du_device_info rzg1_du_r8a77470_info = {
|
||||||
.gen = 2,
|
.gen = 2,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
.channels_mask = BIT(1) | BIT(0),
|
.channels_mask = BIT(1) | BIT(0),
|
||||||
@@ -105,7 +109,8 @@ static const struct rcar_du_device_info rzg1_du_r8a77470_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a774a1_info = {
|
static const struct rcar_du_device_info rcar_du_r8a774a1_info = {
|
||||||
.gen = 3,
|
.gen = 3,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_VSP1_SOURCE
|
| RCAR_DU_FEATURE_VSP1_SOURCE
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
@@ -134,7 +139,8 @@ static const struct rcar_du_device_info rcar_du_r8a774a1_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a774b1_info = {
|
static const struct rcar_du_device_info rcar_du_r8a774b1_info = {
|
||||||
.gen = 3,
|
.gen = 3,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_VSP1_SOURCE
|
| RCAR_DU_FEATURE_VSP1_SOURCE
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
@@ -163,7 +169,8 @@ static const struct rcar_du_device_info rcar_du_r8a774b1_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a774c0_info = {
|
static const struct rcar_du_device_info rcar_du_r8a774c0_info = {
|
||||||
.gen = 3,
|
.gen = 3,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_VSP1_SOURCE,
|
| RCAR_DU_FEATURE_VSP1_SOURCE,
|
||||||
.channels_mask = BIT(1) | BIT(0),
|
.channels_mask = BIT(1) | BIT(0),
|
||||||
.routes = {
|
.routes = {
|
||||||
@@ -189,7 +196,8 @@ static const struct rcar_du_device_info rcar_du_r8a774c0_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a774e1_info = {
|
static const struct rcar_du_device_info rcar_du_r8a774e1_info = {
|
||||||
.gen = 3,
|
.gen = 3,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_VSP1_SOURCE
|
| RCAR_DU_FEATURE_VSP1_SOURCE
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
@@ -239,7 +247,8 @@ static const struct rcar_du_device_info rcar_du_r8a7779_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a7790_info = {
|
static const struct rcar_du_device_info rcar_du_r8a7790_info = {
|
||||||
.gen = 2,
|
.gen = 2,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
.quirks = RCAR_DU_QUIRK_ALIGN_128B,
|
.quirks = RCAR_DU_QUIRK_ALIGN_128B,
|
||||||
@@ -269,7 +278,8 @@ static const struct rcar_du_device_info rcar_du_r8a7790_info = {
|
|||||||
/* M2-W (r8a7791) and M2-N (r8a7793) are identical */
|
/* M2-W (r8a7791) and M2-N (r8a7793) are identical */
|
||||||
static const struct rcar_du_device_info rcar_du_r8a7791_info = {
|
static const struct rcar_du_device_info rcar_du_r8a7791_info = {
|
||||||
.gen = 2,
|
.gen = 2,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
.channels_mask = BIT(1) | BIT(0),
|
.channels_mask = BIT(1) | BIT(0),
|
||||||
@@ -292,7 +302,8 @@ static const struct rcar_du_device_info rcar_du_r8a7791_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a7792_info = {
|
static const struct rcar_du_device_info rcar_du_r8a7792_info = {
|
||||||
.gen = 2,
|
.gen = 2,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
.channels_mask = BIT(1) | BIT(0),
|
.channels_mask = BIT(1) | BIT(0),
|
||||||
@@ -311,7 +322,8 @@ static const struct rcar_du_device_info rcar_du_r8a7792_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a7794_info = {
|
static const struct rcar_du_device_info rcar_du_r8a7794_info = {
|
||||||
.gen = 2,
|
.gen = 2,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
.channels_mask = BIT(1) | BIT(0),
|
.channels_mask = BIT(1) | BIT(0),
|
||||||
@@ -333,7 +345,8 @@ static const struct rcar_du_device_info rcar_du_r8a7794_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a7795_info = {
|
static const struct rcar_du_device_info rcar_du_r8a7795_info = {
|
||||||
.gen = 3,
|
.gen = 3,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_VSP1_SOURCE
|
| RCAR_DU_FEATURE_VSP1_SOURCE
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
@@ -366,7 +379,8 @@ static const struct rcar_du_device_info rcar_du_r8a7795_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a7796_info = {
|
static const struct rcar_du_device_info rcar_du_r8a7796_info = {
|
||||||
.gen = 3,
|
.gen = 3,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_VSP1_SOURCE
|
| RCAR_DU_FEATURE_VSP1_SOURCE
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
@@ -395,7 +409,8 @@ static const struct rcar_du_device_info rcar_du_r8a7796_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a77965_info = {
|
static const struct rcar_du_device_info rcar_du_r8a77965_info = {
|
||||||
.gen = 3,
|
.gen = 3,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_VSP1_SOURCE
|
| RCAR_DU_FEATURE_VSP1_SOURCE
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
@@ -424,7 +439,8 @@ static const struct rcar_du_device_info rcar_du_r8a77965_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a77970_info = {
|
static const struct rcar_du_device_info rcar_du_r8a77970_info = {
|
||||||
.gen = 3,
|
.gen = 3,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_VSP1_SOURCE
|
| RCAR_DU_FEATURE_VSP1_SOURCE
|
||||||
| RCAR_DU_FEATURE_INTERLACED
|
| RCAR_DU_FEATURE_INTERLACED
|
||||||
| RCAR_DU_FEATURE_TVM_SYNC,
|
| RCAR_DU_FEATURE_TVM_SYNC,
|
||||||
@@ -448,7 +464,8 @@ static const struct rcar_du_device_info rcar_du_r8a77970_info = {
|
|||||||
|
|
||||||
static const struct rcar_du_device_info rcar_du_r8a7799x_info = {
|
static const struct rcar_du_device_info rcar_du_r8a7799x_info = {
|
||||||
.gen = 3,
|
.gen = 3,
|
||||||
.features = RCAR_DU_FEATURE_CRTC_IRQ_CLOCK
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_CRTC_CLOCK
|
||||||
| RCAR_DU_FEATURE_VSP1_SOURCE,
|
| RCAR_DU_FEATURE_VSP1_SOURCE,
|
||||||
.channels_mask = BIT(1) | BIT(0),
|
.channels_mask = BIT(1) | BIT(0),
|
||||||
.routes = {
|
.routes = {
|
||||||
@@ -473,6 +490,25 @@ static const struct rcar_du_device_info rcar_du_r8a7799x_info = {
|
|||||||
.lvds_clk_mask = BIT(1) | BIT(0),
|
.lvds_clk_mask = BIT(1) | BIT(0),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct rcar_du_device_info rcar_du_r8a779a0_info = {
|
||||||
|
.gen = 3,
|
||||||
|
.features = RCAR_DU_FEATURE_CRTC_IRQ
|
||||||
|
| RCAR_DU_FEATURE_VSP1_SOURCE,
|
||||||
|
.channels_mask = BIT(1) | BIT(0),
|
||||||
|
.routes = {
|
||||||
|
/* R8A779A0 has two MIPI DSI outputs. */
|
||||||
|
[RCAR_DU_OUTPUT_DSI0] = {
|
||||||
|
.possible_crtcs = BIT(0),
|
||||||
|
.port = 0,
|
||||||
|
},
|
||||||
|
[RCAR_DU_OUTPUT_DSI1] = {
|
||||||
|
.possible_crtcs = BIT(1),
|
||||||
|
.port = 1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
.dsi_clk_mask = BIT(1) | BIT(0),
|
||||||
|
};
|
||||||
|
|
||||||
static const struct of_device_id rcar_du_of_table[] = {
|
static const struct of_device_id rcar_du_of_table[] = {
|
||||||
{ .compatible = "renesas,du-r8a7742", .data = &rcar_du_r8a7790_info },
|
{ .compatible = "renesas,du-r8a7742", .data = &rcar_du_r8a7790_info },
|
||||||
{ .compatible = "renesas,du-r8a7743", .data = &rzg1_du_r8a7743_info },
|
{ .compatible = "renesas,du-r8a7743", .data = &rzg1_du_r8a7743_info },
|
||||||
@@ -497,11 +533,30 @@ static const struct of_device_id rcar_du_of_table[] = {
|
|||||||
{ .compatible = "renesas,du-r8a77980", .data = &rcar_du_r8a77970_info },
|
{ .compatible = "renesas,du-r8a77980", .data = &rcar_du_r8a77970_info },
|
||||||
{ .compatible = "renesas,du-r8a77990", .data = &rcar_du_r8a7799x_info },
|
{ .compatible = "renesas,du-r8a77990", .data = &rcar_du_r8a7799x_info },
|
||||||
{ .compatible = "renesas,du-r8a77995", .data = &rcar_du_r8a7799x_info },
|
{ .compatible = "renesas,du-r8a77995", .data = &rcar_du_r8a7799x_info },
|
||||||
|
{ .compatible = "renesas,du-r8a779a0", .data = &rcar_du_r8a779a0_info },
|
||||||
{ }
|
{ }
|
||||||
};
|
};
|
||||||
|
|
||||||
MODULE_DEVICE_TABLE(of, rcar_du_of_table);
|
MODULE_DEVICE_TABLE(of, rcar_du_of_table);
|
||||||
|
|
||||||
|
const char *rcar_du_output_name(enum rcar_du_output output)
|
||||||
|
{
|
||||||
|
static const char * const names[] = {
|
||||||
|
[RCAR_DU_OUTPUT_DPAD0] = "DPAD0",
|
||||||
|
[RCAR_DU_OUTPUT_DPAD1] = "DPAD1",
|
||||||
|
[RCAR_DU_OUTPUT_LVDS0] = "LVDS0",
|
||||||
|
[RCAR_DU_OUTPUT_LVDS1] = "LVDS1",
|
||||||
|
[RCAR_DU_OUTPUT_HDMI0] = "HDMI0",
|
||||||
|
[RCAR_DU_OUTPUT_HDMI1] = "HDMI1",
|
||||||
|
[RCAR_DU_OUTPUT_TCON] = "TCON",
|
||||||
|
};
|
||||||
|
|
||||||
|
if (output >= ARRAY_SIZE(names) || !names[output])
|
||||||
|
return "UNKNOWN";
|
||||||
|
|
||||||
|
return names[output];
|
||||||
|
}
|
||||||
|
|
||||||
/* -----------------------------------------------------------------------------
|
/* -----------------------------------------------------------------------------
|
||||||
* DRM operations
|
* DRM operations
|
||||||
*/
|
*/
|
||||||
@@ -510,7 +565,11 @@ DEFINE_DRM_GEM_CMA_FOPS(rcar_du_fops);
|
|||||||
|
|
||||||
static const struct drm_driver rcar_du_driver = {
|
static const struct drm_driver rcar_du_driver = {
|
||||||
.driver_features = DRIVER_GEM | DRIVER_MODESET | DRIVER_ATOMIC,
|
.driver_features = DRIVER_GEM | DRIVER_MODESET | DRIVER_ATOMIC,
|
||||||
DRM_GEM_CMA_DRIVER_OPS_WITH_DUMB_CREATE(rcar_du_dumb_create),
|
.dumb_create = rcar_du_dumb_create,
|
||||||
|
.prime_handle_to_fd = drm_gem_prime_handle_to_fd,
|
||||||
|
.prime_fd_to_handle = drm_gem_prime_fd_to_handle,
|
||||||
|
.gem_prime_import_sg_table = rcar_du_gem_prime_import_sg_table,
|
||||||
|
.gem_prime_mmap = drm_gem_prime_mmap,
|
||||||
.fops = &rcar_du_fops,
|
.fops = &rcar_du_fops,
|
||||||
.name = "rcar-du",
|
.name = "rcar-du",
|
||||||
.desc = "Renesas R-Car Display Unit",
|
.desc = "Renesas R-Car Display Unit",
|
||||||
@@ -570,7 +629,7 @@ static void rcar_du_shutdown(struct platform_device *pdev)
|
|||||||
static int rcar_du_probe(struct platform_device *pdev)
|
static int rcar_du_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct rcar_du_device *rcdu;
|
struct rcar_du_device *rcdu;
|
||||||
struct resource *mem;
|
unsigned int mask;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
/* Allocate and initialize the R-Car device structure. */
|
/* Allocate and initialize the R-Car device structure. */
|
||||||
@@ -585,11 +644,20 @@ static int rcar_du_probe(struct platform_device *pdev)
|
|||||||
platform_set_drvdata(pdev, rcdu);
|
platform_set_drvdata(pdev, rcdu);
|
||||||
|
|
||||||
/* I/O resources */
|
/* I/O resources */
|
||||||
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
rcdu->mmio = devm_platform_ioremap_resource(pdev, 0);
|
||||||
rcdu->mmio = devm_ioremap_resource(&pdev->dev, mem);
|
|
||||||
if (IS_ERR(rcdu->mmio))
|
if (IS_ERR(rcdu->mmio))
|
||||||
return PTR_ERR(rcdu->mmio);
|
return PTR_ERR(rcdu->mmio);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set the DMA coherent mask to reflect the DU 32-bit DMA address space
|
||||||
|
* limitations. When sourcing frames from a VSP the DU doesn't perform
|
||||||
|
* any memory access so set the mask to 40 bits to accept all buffers.
|
||||||
|
*/
|
||||||
|
mask = rcar_du_has(rcdu, RCAR_DU_FEATURE_VSP1_SOURCE) ? 40 : 32;
|
||||||
|
ret = dma_coerce_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(mask));
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
/* DRM/KMS objects */
|
/* DRM/KMS objects */
|
||||||
ret = rcar_du_modeset_init(rcdu);
|
ret = rcar_du_modeset_init(rcdu);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
|
|||||||
@@ -26,13 +26,27 @@ struct drm_bridge;
|
|||||||
struct drm_property;
|
struct drm_property;
|
||||||
struct rcar_du_device;
|
struct rcar_du_device;
|
||||||
|
|
||||||
#define RCAR_DU_FEATURE_CRTC_IRQ_CLOCK BIT(0) /* Per-CRTC IRQ and clock */
|
#define RCAR_DU_FEATURE_CRTC_IRQ BIT(0) /* Per-CRTC IRQ */
|
||||||
#define RCAR_DU_FEATURE_VSP1_SOURCE BIT(1) /* Has inputs from VSP1 */
|
#define RCAR_DU_FEATURE_CRTC_CLOCK BIT(1) /* Per-CRTC clock */
|
||||||
#define RCAR_DU_FEATURE_INTERLACED BIT(2) /* HW supports interlaced */
|
#define RCAR_DU_FEATURE_VSP1_SOURCE BIT(2) /* Has inputs from VSP1 */
|
||||||
#define RCAR_DU_FEATURE_TVM_SYNC BIT(3) /* Has TV switch/sync modes */
|
#define RCAR_DU_FEATURE_INTERLACED BIT(3) /* HW supports interlaced */
|
||||||
|
#define RCAR_DU_FEATURE_TVM_SYNC BIT(4) /* Has TV switch/sync modes */
|
||||||
|
|
||||||
#define RCAR_DU_QUIRK_ALIGN_128B BIT(0) /* Align pitches to 128 bytes */
|
#define RCAR_DU_QUIRK_ALIGN_128B BIT(0) /* Align pitches to 128 bytes */
|
||||||
|
|
||||||
|
enum rcar_du_output {
|
||||||
|
RCAR_DU_OUTPUT_DPAD0,
|
||||||
|
RCAR_DU_OUTPUT_DPAD1,
|
||||||
|
RCAR_DU_OUTPUT_DSI0,
|
||||||
|
RCAR_DU_OUTPUT_DSI1,
|
||||||
|
RCAR_DU_OUTPUT_HDMI0,
|
||||||
|
RCAR_DU_OUTPUT_HDMI1,
|
||||||
|
RCAR_DU_OUTPUT_LVDS0,
|
||||||
|
RCAR_DU_OUTPUT_LVDS1,
|
||||||
|
RCAR_DU_OUTPUT_TCON,
|
||||||
|
RCAR_DU_OUTPUT_MAX,
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* struct rcar_du_output_routing - Output routing specification
|
* struct rcar_du_output_routing - Output routing specification
|
||||||
* @possible_crtcs: bitmask of possible CRTCs for the output
|
* @possible_crtcs: bitmask of possible CRTCs for the output
|
||||||
@@ -56,6 +70,7 @@ struct rcar_du_output_routing {
|
|||||||
* @routes: array of CRTC to output routes, indexed by output (RCAR_DU_OUTPUT_*)
|
* @routes: array of CRTC to output routes, indexed by output (RCAR_DU_OUTPUT_*)
|
||||||
* @num_lvds: number of internal LVDS encoders
|
* @num_lvds: number of internal LVDS encoders
|
||||||
* @dpll_mask: bit mask of DU channels equipped with a DPLL
|
* @dpll_mask: bit mask of DU channels equipped with a DPLL
|
||||||
|
* @dsi_clk_mask: bitmask of channels that can use the DSI clock as dot clock
|
||||||
* @lvds_clk_mask: bitmask of channels that can use the LVDS clock as dot clock
|
* @lvds_clk_mask: bitmask of channels that can use the LVDS clock as dot clock
|
||||||
*/
|
*/
|
||||||
struct rcar_du_device_info {
|
struct rcar_du_device_info {
|
||||||
@@ -66,6 +81,7 @@ struct rcar_du_device_info {
|
|||||||
struct rcar_du_output_routing routes[RCAR_DU_OUTPUT_MAX];
|
struct rcar_du_output_routing routes[RCAR_DU_OUTPUT_MAX];
|
||||||
unsigned int num_lvds;
|
unsigned int num_lvds;
|
||||||
unsigned int dpll_mask;
|
unsigned int dpll_mask;
|
||||||
|
unsigned int dsi_clk_mask;
|
||||||
unsigned int lvds_clk_mask;
|
unsigned int lvds_clk_mask;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -126,4 +142,6 @@ static inline void rcar_du_write(struct rcar_du_device *rcdu, u32 reg, u32 data)
|
|||||||
iowrite32(data, rcdu->mmio + reg);
|
iowrite32(data, rcdu->mmio + reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const char *rcar_du_output_name(enum rcar_du_output output);
|
||||||
|
|
||||||
#endif /* __RCAR_DU_DRV_H__ */
|
#endif /* __RCAR_DU_DRV_H__ */
|
||||||
|
|||||||
@@ -86,17 +86,25 @@ int rcar_du_encoder_init(struct rcar_du_device *rcdu,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Create and initialize the encoder. On Gen3 skip the LVDS1 output if
|
* Create and initialize the encoder. On Gen3, skip the LVDS1 output if
|
||||||
* the LVDS1 encoder is used as a companion for LVDS0 in dual-link
|
* the LVDS1 encoder is used as a companion for LVDS0 in dual-link
|
||||||
* mode.
|
* mode, or any LVDS output if it isn't connected. The latter may happen
|
||||||
|
* on D3 or E3 as the LVDS encoders are needed to provide the pixel
|
||||||
|
* clock to the DU, even when the LVDS outputs are not used.
|
||||||
*/
|
*/
|
||||||
if (rcdu->info->gen >= 3 && output == RCAR_DU_OUTPUT_LVDS1) {
|
if (rcdu->info->gen >= 3) {
|
||||||
if (rcar_lvds_dual_link(bridge))
|
if (output == RCAR_DU_OUTPUT_LVDS1 &&
|
||||||
|
rcar_lvds_dual_link(bridge))
|
||||||
|
return -ENOLINK;
|
||||||
|
|
||||||
|
if ((output == RCAR_DU_OUTPUT_LVDS0 ||
|
||||||
|
output == RCAR_DU_OUTPUT_LVDS1) &&
|
||||||
|
!rcar_lvds_is_connected(bridge))
|
||||||
return -ENOLINK;
|
return -ENOLINK;
|
||||||
}
|
}
|
||||||
|
|
||||||
dev_dbg(rcdu->dev, "initializing encoder %pOF for output %u\n",
|
dev_dbg(rcdu->dev, "initializing encoder %pOF for output %s\n",
|
||||||
enc_node, output);
|
enc_node, rcar_du_output_name(output));
|
||||||
|
|
||||||
renc = drmm_encoder_alloc(&rcdu->ddev, struct rcar_du_encoder, base,
|
renc = drmm_encoder_alloc(&rcdu->ddev, struct rcar_du_encoder, base,
|
||||||
&rcar_du_encoder_funcs, DRM_MODE_ENCODER_NONE,
|
&rcar_du_encoder_funcs, DRM_MODE_ENCODER_NONE,
|
||||||
@@ -110,8 +118,9 @@ int rcar_du_encoder_init(struct rcar_du_device *rcdu,
|
|||||||
ret = drm_bridge_attach(&renc->base, bridge, NULL,
|
ret = drm_bridge_attach(&renc->base, bridge, NULL,
|
||||||
DRM_BRIDGE_ATTACH_NO_CONNECTOR);
|
DRM_BRIDGE_ATTACH_NO_CONNECTOR);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
dev_err(rcdu->dev, "failed to attach bridge for output %u\n",
|
dev_err(rcdu->dev,
|
||||||
output);
|
"failed to attach bridge %pOF for output %s (%d)\n",
|
||||||
|
bridge->of_node, rcar_du_output_name(output), ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -119,7 +128,8 @@ int rcar_du_encoder_init(struct rcar_du_device *rcdu,
|
|||||||
connector = drm_bridge_connector_init(&rcdu->ddev, &renc->base);
|
connector = drm_bridge_connector_init(&rcdu->ddev, &renc->base);
|
||||||
if (IS_ERR(connector)) {
|
if (IS_ERR(connector)) {
|
||||||
dev_err(rcdu->dev,
|
dev_err(rcdu->dev,
|
||||||
"failed to created connector for output %u\n", output);
|
"failed to created connector for output %s (%ld)\n",
|
||||||
|
rcar_du_output_name(output), PTR_ERR(connector));
|
||||||
return PTR_ERR(connector);
|
return PTR_ERR(connector);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -122,10 +122,12 @@ static void rcar_du_group_setup_didsr(struct rcar_du_group *rgrp)
|
|||||||
didsr = DIDSR_CODE;
|
didsr = DIDSR_CODE;
|
||||||
for (i = 0; i < num_crtcs; ++i, ++rcrtc) {
|
for (i = 0; i < num_crtcs; ++i, ++rcrtc) {
|
||||||
if (rcdu->info->lvds_clk_mask & BIT(rcrtc->index))
|
if (rcdu->info->lvds_clk_mask & BIT(rcrtc->index))
|
||||||
didsr |= DIDSR_LCDS_LVDS0(i)
|
didsr |= DIDSR_LDCS_LVDS0(i)
|
||||||
| DIDSR_PDCS_CLK(i, 0);
|
| DIDSR_PDCS_CLK(i, 0);
|
||||||
|
else if (rcdu->info->dsi_clk_mask & BIT(rcrtc->index))
|
||||||
|
didsr |= DIDSR_LDCS_DSI(i);
|
||||||
else
|
else
|
||||||
didsr |= DIDSR_LCDS_DCLKIN(i)
|
didsr |= DIDSR_LDCS_DCLKIN(i)
|
||||||
| DIDSR_PDCS_CLK(i, 0);
|
| DIDSR_PDCS_CLK(i, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -19,6 +19,7 @@
|
|||||||
#include <drm/drm_vblank.h>
|
#include <drm/drm_vblank.h>
|
||||||
|
|
||||||
#include <linux/device.h>
|
#include <linux/device.h>
|
||||||
|
#include <linux/dma-buf.h>
|
||||||
#include <linux/of_graph.h>
|
#include <linux/of_graph.h>
|
||||||
#include <linux/of_platform.h>
|
#include <linux/of_platform.h>
|
||||||
#include <linux/wait.h>
|
#include <linux/wait.h>
|
||||||
@@ -325,6 +326,51 @@ const struct rcar_du_format_info *rcar_du_format_info(u32 fourcc)
|
|||||||
* Frame buffer
|
* Frame buffer
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
static const struct drm_gem_object_funcs rcar_du_gem_funcs = {
|
||||||
|
.free = drm_gem_cma_free_object,
|
||||||
|
.print_info = drm_gem_cma_print_info,
|
||||||
|
.get_sg_table = drm_gem_cma_get_sg_table,
|
||||||
|
.vmap = drm_gem_cma_vmap,
|
||||||
|
.mmap = drm_gem_cma_mmap,
|
||||||
|
.vm_ops = &drm_gem_cma_vm_ops,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct drm_gem_object *rcar_du_gem_prime_import_sg_table(struct drm_device *dev,
|
||||||
|
struct dma_buf_attachment *attach,
|
||||||
|
struct sg_table *sgt)
|
||||||
|
{
|
||||||
|
struct rcar_du_device *rcdu = to_rcar_du_device(dev);
|
||||||
|
struct drm_gem_cma_object *cma_obj;
|
||||||
|
struct drm_gem_object *gem_obj;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if (!rcar_du_has(rcdu, RCAR_DU_FEATURE_VSP1_SOURCE))
|
||||||
|
return drm_gem_cma_prime_import_sg_table(dev, attach, sgt);
|
||||||
|
|
||||||
|
/* Create a CMA GEM buffer. */
|
||||||
|
cma_obj = kzalloc(sizeof(*cma_obj), GFP_KERNEL);
|
||||||
|
if (!cma_obj)
|
||||||
|
return ERR_PTR(-ENOMEM);
|
||||||
|
|
||||||
|
gem_obj = &cma_obj->base;
|
||||||
|
gem_obj->funcs = &rcar_du_gem_funcs;
|
||||||
|
|
||||||
|
drm_gem_private_object_init(dev, gem_obj, attach->dmabuf->size);
|
||||||
|
cma_obj->map_noncoherent = false;
|
||||||
|
|
||||||
|
ret = drm_gem_create_mmap_offset(gem_obj);
|
||||||
|
if (ret) {
|
||||||
|
drm_gem_object_release(gem_obj);
|
||||||
|
kfree(cma_obj);
|
||||||
|
return ERR_PTR(ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
cma_obj->paddr = 0;
|
||||||
|
cma_obj->sgt = sgt;
|
||||||
|
|
||||||
|
return gem_obj;
|
||||||
|
}
|
||||||
|
|
||||||
int rcar_du_dumb_create(struct drm_file *file, struct drm_device *dev,
|
int rcar_du_dumb_create(struct drm_file *file, struct drm_device *dev,
|
||||||
struct drm_mode_create_dumb *args)
|
struct drm_mode_create_dumb *args)
|
||||||
{
|
{
|
||||||
@@ -513,8 +559,8 @@ static int rcar_du_encoders_init_one(struct rcar_du_device *rcdu,
|
|||||||
ret = rcar_du_encoder_init(rcdu, output, entity);
|
ret = rcar_du_encoder_init(rcdu, output, entity);
|
||||||
if (ret && ret != -EPROBE_DEFER && ret != -ENOLINK)
|
if (ret && ret != -EPROBE_DEFER && ret != -ENOLINK)
|
||||||
dev_warn(rcdu->dev,
|
dev_warn(rcdu->dev,
|
||||||
"failed to initialize encoder %pOF on output %u (%d), skipping\n",
|
"failed to initialize encoder %pOF on output %s (%d), skipping\n",
|
||||||
entity, output, ret);
|
entity, rcar_du_output_name(output), ret);
|
||||||
|
|
||||||
of_node_put(entity);
|
of_node_put(entity);
|
||||||
|
|
||||||
|
|||||||
@@ -12,10 +12,13 @@
|
|||||||
|
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
|
|
||||||
|
struct dma_buf_attachment;
|
||||||
struct drm_file;
|
struct drm_file;
|
||||||
struct drm_device;
|
struct drm_device;
|
||||||
|
struct drm_gem_object;
|
||||||
struct drm_mode_create_dumb;
|
struct drm_mode_create_dumb;
|
||||||
struct rcar_du_device;
|
struct rcar_du_device;
|
||||||
|
struct sg_table;
|
||||||
|
|
||||||
struct rcar_du_format_info {
|
struct rcar_du_format_info {
|
||||||
u32 fourcc;
|
u32 fourcc;
|
||||||
@@ -34,4 +37,8 @@ int rcar_du_modeset_init(struct rcar_du_device *rcdu);
|
|||||||
int rcar_du_dumb_create(struct drm_file *file, struct drm_device *dev,
|
int rcar_du_dumb_create(struct drm_file *file, struct drm_device *dev,
|
||||||
struct drm_mode_create_dumb *args);
|
struct drm_mode_create_dumb *args);
|
||||||
|
|
||||||
|
struct drm_gem_object *rcar_du_gem_prime_import_sg_table(struct drm_device *dev,
|
||||||
|
struct dma_buf_attachment *attach,
|
||||||
|
struct sg_table *sgt);
|
||||||
|
|
||||||
#endif /* __RCAR_DU_KMS_H__ */
|
#endif /* __RCAR_DU_KMS_H__ */
|
||||||
|
|||||||
@@ -257,10 +257,11 @@
|
|||||||
|
|
||||||
#define DIDSR 0x20028
|
#define DIDSR 0x20028
|
||||||
#define DIDSR_CODE (0x7790 << 16)
|
#define DIDSR_CODE (0x7790 << 16)
|
||||||
#define DIDSR_LCDS_DCLKIN(n) (0 << (8 + (n) * 2))
|
#define DIDSR_LDCS_DCLKIN(n) (0 << (8 + (n) * 2))
|
||||||
#define DIDSR_LCDS_LVDS0(n) (2 << (8 + (n) * 2))
|
#define DIDSR_LDCS_DSI(n) (2 << (8 + (n) * 2)) /* V3U only */
|
||||||
#define DIDSR_LCDS_LVDS1(n) (3 << (8 + (n) * 2))
|
#define DIDSR_LDCS_LVDS0(n) (2 << (8 + (n) * 2))
|
||||||
#define DIDSR_LCDS_MASK(n) (3 << (8 + (n) * 2))
|
#define DIDSR_LDCS_LVDS1(n) (3 << (8 + (n) * 2))
|
||||||
|
#define DIDSR_LDCS_MASK(n) (3 << (8 + (n) * 2))
|
||||||
#define DIDSR_PDCS_CLK(n, clk) (clk << ((n) * 2))
|
#define DIDSR_PDCS_CLK(n, clk) (clk << ((n) * 2))
|
||||||
#define DIDSR_PDCS_MASK(n) (3 << ((n) * 2))
|
#define DIDSR_PDCS_MASK(n) (3 << ((n) * 2))
|
||||||
|
|
||||||
|
|||||||
@@ -187,17 +187,43 @@ int rcar_du_vsp_map_fb(struct rcar_du_vsp *vsp, struct drm_framebuffer *fb,
|
|||||||
struct sg_table sg_tables[3])
|
struct sg_table sg_tables[3])
|
||||||
{
|
{
|
||||||
struct rcar_du_device *rcdu = vsp->dev;
|
struct rcar_du_device *rcdu = vsp->dev;
|
||||||
unsigned int i;
|
unsigned int i, j;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
for (i = 0; i < fb->format->num_planes; ++i) {
|
for (i = 0; i < fb->format->num_planes; ++i) {
|
||||||
struct drm_gem_cma_object *gem = drm_fb_cma_get_gem_obj(fb, i);
|
struct drm_gem_cma_object *gem = drm_fb_cma_get_gem_obj(fb, i);
|
||||||
struct sg_table *sgt = &sg_tables[i];
|
struct sg_table *sgt = &sg_tables[i];
|
||||||
|
|
||||||
ret = dma_get_sgtable(rcdu->dev, sgt, gem->vaddr, gem->paddr,
|
if (gem->sgt) {
|
||||||
gem->base.size);
|
struct scatterlist *src;
|
||||||
if (ret)
|
struct scatterlist *dst;
|
||||||
goto fail;
|
|
||||||
|
/*
|
||||||
|
* If the GEM buffer has a scatter gather table, it has
|
||||||
|
* been imported from a dma-buf and has no physical
|
||||||
|
* address as it might not be physically contiguous.
|
||||||
|
* Copy the original scatter gather table to map it to
|
||||||
|
* the VSP.
|
||||||
|
*/
|
||||||
|
ret = sg_alloc_table(sgt, gem->sgt->orig_nents,
|
||||||
|
GFP_KERNEL);
|
||||||
|
if (ret)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
src = gem->sgt->sgl;
|
||||||
|
dst = sgt->sgl;
|
||||||
|
for (j = 0; j < gem->sgt->orig_nents; ++j) {
|
||||||
|
sg_set_page(dst, sg_page(src), src->length,
|
||||||
|
src->offset);
|
||||||
|
src = sg_next(src);
|
||||||
|
dst = sg_next(dst);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ret = dma_get_sgtable(rcdu->dev, sgt, gem->vaddr,
|
||||||
|
gem->paddr, gem->base.size);
|
||||||
|
if (ret)
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
ret = vsp1_du_map_sg(vsp->vsp, sgt);
|
ret = vsp1_du_map_sg(vsp->vsp, sgt);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
|||||||
@@ -576,6 +576,9 @@ static int rcar_lvds_attach(struct drm_bridge *bridge,
|
|||||||
{
|
{
|
||||||
struct rcar_lvds *lvds = bridge_to_rcar_lvds(bridge);
|
struct rcar_lvds *lvds = bridge_to_rcar_lvds(bridge);
|
||||||
|
|
||||||
|
if (!lvds->next_bridge)
|
||||||
|
return 0;
|
||||||
|
|
||||||
return drm_bridge_attach(bridge->encoder, lvds->next_bridge, bridge,
|
return drm_bridge_attach(bridge->encoder, lvds->next_bridge, bridge,
|
||||||
flags);
|
flags);
|
||||||
}
|
}
|
||||||
@@ -598,6 +601,14 @@ bool rcar_lvds_dual_link(struct drm_bridge *bridge)
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(rcar_lvds_dual_link);
|
EXPORT_SYMBOL_GPL(rcar_lvds_dual_link);
|
||||||
|
|
||||||
|
bool rcar_lvds_is_connected(struct drm_bridge *bridge)
|
||||||
|
{
|
||||||
|
struct rcar_lvds *lvds = bridge_to_rcar_lvds(bridge);
|
||||||
|
|
||||||
|
return lvds->next_bridge != NULL;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(rcar_lvds_is_connected);
|
||||||
|
|
||||||
/* -----------------------------------------------------------------------------
|
/* -----------------------------------------------------------------------------
|
||||||
* Probe & Remove
|
* Probe & Remove
|
||||||
*/
|
*/
|
||||||
@@ -802,7 +813,6 @@ static int rcar_lvds_probe(struct platform_device *pdev)
|
|||||||
{
|
{
|
||||||
const struct soc_device_attribute *attr;
|
const struct soc_device_attribute *attr;
|
||||||
struct rcar_lvds *lvds;
|
struct rcar_lvds *lvds;
|
||||||
struct resource *mem;
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
lvds = devm_kzalloc(&pdev->dev, sizeof(*lvds), GFP_KERNEL);
|
lvds = devm_kzalloc(&pdev->dev, sizeof(*lvds), GFP_KERNEL);
|
||||||
@@ -825,8 +835,7 @@ static int rcar_lvds_probe(struct platform_device *pdev)
|
|||||||
lvds->bridge.funcs = &rcar_lvds_bridge_ops;
|
lvds->bridge.funcs = &rcar_lvds_bridge_ops;
|
||||||
lvds->bridge.of_node = pdev->dev.of_node;
|
lvds->bridge.of_node = pdev->dev.of_node;
|
||||||
|
|
||||||
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
lvds->mmio = devm_platform_ioremap_resource(pdev, 0);
|
||||||
lvds->mmio = devm_ioremap_resource(&pdev->dev, mem);
|
|
||||||
if (IS_ERR(lvds->mmio))
|
if (IS_ERR(lvds->mmio))
|
||||||
return PTR_ERR(lvds->mmio);
|
return PTR_ERR(lvds->mmio);
|
||||||
|
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ struct drm_bridge;
|
|||||||
int rcar_lvds_clk_enable(struct drm_bridge *bridge, unsigned long freq);
|
int rcar_lvds_clk_enable(struct drm_bridge *bridge, unsigned long freq);
|
||||||
void rcar_lvds_clk_disable(struct drm_bridge *bridge);
|
void rcar_lvds_clk_disable(struct drm_bridge *bridge);
|
||||||
bool rcar_lvds_dual_link(struct drm_bridge *bridge);
|
bool rcar_lvds_dual_link(struct drm_bridge *bridge);
|
||||||
|
bool rcar_lvds_is_connected(struct drm_bridge *bridge);
|
||||||
#else
|
#else
|
||||||
static inline int rcar_lvds_clk_enable(struct drm_bridge *bridge,
|
static inline int rcar_lvds_clk_enable(struct drm_bridge *bridge,
|
||||||
unsigned long freq)
|
unsigned long freq)
|
||||||
@@ -27,6 +28,10 @@ static inline bool rcar_lvds_dual_link(struct drm_bridge *bridge)
|
|||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
static inline bool rcar_lvds_is_connected(struct drm_bridge *bridge)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
#endif /* CONFIG_DRM_RCAR_LVDS */
|
#endif /* CONFIG_DRM_RCAR_LVDS */
|
||||||
|
|
||||||
#endif /* __RCAR_LVDS_H__ */
|
#endif /* __RCAR_LVDS_H__ */
|
||||||
|
|||||||
@@ -192,7 +192,6 @@ static int shmob_drm_probe(struct platform_device *pdev)
|
|||||||
struct shmob_drm_platform_data *pdata = pdev->dev.platform_data;
|
struct shmob_drm_platform_data *pdata = pdev->dev.platform_data;
|
||||||
struct shmob_drm_device *sdev;
|
struct shmob_drm_device *sdev;
|
||||||
struct drm_device *ddev;
|
struct drm_device *ddev;
|
||||||
struct resource *res;
|
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
@@ -215,8 +214,7 @@ static int shmob_drm_probe(struct platform_device *pdev)
|
|||||||
|
|
||||||
platform_set_drvdata(pdev, sdev);
|
platform_set_drvdata(pdev, sdev);
|
||||||
|
|
||||||
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
sdev->mmio = devm_platform_ioremap_resource(pdev, 0);
|
||||||
sdev->mmio = devm_ioremap_resource(&pdev->dev, res);
|
|
||||||
if (IS_ERR(sdev->mmio))
|
if (IS_ERR(sdev->mmio))
|
||||||
return PTR_ERR(sdev->mmio);
|
return PTR_ERR(sdev->mmio);
|
||||||
|
|
||||||
|
|||||||
@@ -927,12 +927,12 @@ static void sti_hqvdp_start_xp70(struct sti_hqvdp *hqvdp)
|
|||||||
|
|
||||||
header = (struct fw_header *)firmware->data;
|
header = (struct fw_header *)firmware->data;
|
||||||
if (firmware->size < sizeof(*header)) {
|
if (firmware->size < sizeof(*header)) {
|
||||||
DRM_ERROR("Invalid firmware size (%d)\n", firmware->size);
|
DRM_ERROR("Invalid firmware size (%zu)\n", firmware->size);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
if ((sizeof(*header) + header->rd_size + header->wr_size +
|
if ((sizeof(*header) + header->rd_size + header->wr_size +
|
||||||
header->pmem_size + header->dmem_size) != firmware->size) {
|
header->pmem_size + header->dmem_size) != firmware->size) {
|
||||||
DRM_ERROR("Invalid fmw structure (%d+%d+%d+%d+%d != %d)\n",
|
DRM_ERROR("Invalid fmw structure (%zu+%d+%d+%d+%d != %zu)\n",
|
||||||
sizeof(*header), header->rd_size, header->wr_size,
|
sizeof(*header), header->rd_size, header->wr_size,
|
||||||
header->pmem_size, header->dmem_size,
|
header->pmem_size, header->dmem_size,
|
||||||
firmware->size);
|
firmware->size);
|
||||||
|
|||||||
Reference in New Issue
Block a user