|  | /* | 
|  | * rcar_du_plane.c  --  R-Car Display Unit Planes | 
|  | * | 
|  | * Copyright (C) 2013 Renesas Corporation | 
|  | * | 
|  | * Contact: Laurent Pinchart (laurent.pinchart@ideasonboard.com) | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or modify | 
|  | * it under the terms of the GNU General Public License as published by | 
|  | * the Free Software Foundation; either version 2 of the License, or | 
|  | * (at your option) any later version. | 
|  | */ | 
|  |  | 
|  | #include <drm/drmP.h> | 
|  | #include <drm/drm_crtc.h> | 
|  | #include <drm/drm_crtc_helper.h> | 
|  | #include <drm/drm_fb_cma_helper.h> | 
|  | #include <drm/drm_gem_cma_helper.h> | 
|  |  | 
|  | #include "rcar_du_drv.h" | 
|  | #include "rcar_du_kms.h" | 
|  | #include "rcar_du_plane.h" | 
|  | #include "rcar_du_regs.h" | 
|  |  | 
|  | #define RCAR_DU_COLORKEY_NONE		(0 << 24) | 
|  | #define RCAR_DU_COLORKEY_SOURCE		(1 << 24) | 
|  | #define RCAR_DU_COLORKEY_MASK		(1 << 24) | 
|  |  | 
|  | struct rcar_du_kms_plane { | 
|  | struct drm_plane plane; | 
|  | struct rcar_du_plane *hwplane; | 
|  | }; | 
|  |  | 
|  | static inline struct rcar_du_plane *to_rcar_plane(struct drm_plane *plane) | 
|  | { | 
|  | return container_of(plane, struct rcar_du_kms_plane, plane)->hwplane; | 
|  | } | 
|  |  | 
|  | static u32 rcar_du_plane_read(struct rcar_du_device *rcdu, | 
|  | unsigned int index, u32 reg) | 
|  | { | 
|  | return rcar_du_read(rcdu, index * PLANE_OFF + reg); | 
|  | } | 
|  |  | 
|  | static void rcar_du_plane_write(struct rcar_du_device *rcdu, | 
|  | unsigned int index, u32 reg, u32 data) | 
|  | { | 
|  | rcar_du_write(rcdu, index * PLANE_OFF + reg, data); | 
|  | } | 
|  |  | 
|  | int rcar_du_plane_reserve(struct rcar_du_plane *plane, | 
|  | const struct rcar_du_format_info *format) | 
|  | { | 
|  | struct rcar_du_device *rcdu = plane->dev; | 
|  | unsigned int i; | 
|  | int ret = -EBUSY; | 
|  |  | 
|  | mutex_lock(&rcdu->planes.lock); | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(rcdu->planes.planes); ++i) { | 
|  | if (!(rcdu->planes.free & (1 << i))) | 
|  | continue; | 
|  |  | 
|  | if (format->planes == 1 || | 
|  | rcdu->planes.free & (1 << ((i + 1) % 8))) | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (i == ARRAY_SIZE(rcdu->planes.planes)) | 
|  | goto done; | 
|  |  | 
|  | rcdu->planes.free &= ~(1 << i); | 
|  | if (format->planes == 2) | 
|  | rcdu->planes.free &= ~(1 << ((i + 1) % 8)); | 
|  |  | 
|  | plane->hwindex = i; | 
|  |  | 
|  | ret = 0; | 
|  |  | 
|  | done: | 
|  | mutex_unlock(&rcdu->planes.lock); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | void rcar_du_plane_release(struct rcar_du_plane *plane) | 
|  | { | 
|  | struct rcar_du_device *rcdu = plane->dev; | 
|  |  | 
|  | if (plane->hwindex == -1) | 
|  | return; | 
|  |  | 
|  | mutex_lock(&rcdu->planes.lock); | 
|  | rcdu->planes.free |= 1 << plane->hwindex; | 
|  | if (plane->format->planes == 2) | 
|  | rcdu->planes.free |= 1 << ((plane->hwindex + 1) % 8); | 
|  | mutex_unlock(&rcdu->planes.lock); | 
|  |  | 
|  | plane->hwindex = -1; | 
|  | } | 
|  |  | 
|  | void rcar_du_plane_update_base(struct rcar_du_plane *plane) | 
|  | { | 
|  | struct rcar_du_device *rcdu = plane->dev; | 
|  | unsigned int index = plane->hwindex; | 
|  |  | 
|  | /* According to the datasheet the Y position is expressed in raster line | 
|  | * units. However, 32bpp formats seem to require a doubled Y position | 
|  | * value. Similarly, for the second plane, NV12 and NV21 formats seem to | 
|  | * require a halved Y position value. | 
|  | */ | 
|  | rcar_du_plane_write(rcdu, index, PnSPXR, plane->src_x); | 
|  | rcar_du_plane_write(rcdu, index, PnSPYR, plane->src_y * | 
|  | (plane->format->bpp == 32 ? 2 : 1)); | 
|  | rcar_du_plane_write(rcdu, index, PnDSA0R, plane->dma[0]); | 
|  |  | 
|  | if (plane->format->planes == 2) { | 
|  | index = (index + 1) % 8; | 
|  |  | 
|  | rcar_du_plane_write(rcdu, index, PnSPXR, plane->src_x); | 
|  | rcar_du_plane_write(rcdu, index, PnSPYR, plane->src_y * | 
|  | (plane->format->bpp == 16 ? 2 : 1) / 2); | 
|  | rcar_du_plane_write(rcdu, index, PnDSA0R, plane->dma[1]); | 
|  | } | 
|  | } | 
|  |  | 
|  | void rcar_du_plane_compute_base(struct rcar_du_plane *plane, | 
|  | struct drm_framebuffer *fb) | 
|  | { | 
|  | struct drm_gem_cma_object *gem; | 
|  |  | 
|  | gem = drm_fb_cma_get_gem_obj(fb, 0); | 
|  | plane->dma[0] = gem->paddr + fb->offsets[0]; | 
|  |  | 
|  | if (plane->format->planes == 2) { | 
|  | gem = drm_fb_cma_get_gem_obj(fb, 1); | 
|  | plane->dma[1] = gem->paddr + fb->offsets[1]; | 
|  | } | 
|  | } | 
|  |  | 
|  | static void rcar_du_plane_setup_mode(struct rcar_du_plane *plane, | 
|  | unsigned int index) | 
|  | { | 
|  | struct rcar_du_device *rcdu = plane->dev; | 
|  | u32 colorkey; | 
|  | u32 pnmr; | 
|  |  | 
|  | /* The PnALPHAR register controls alpha-blending in 16bpp formats | 
|  | * (ARGB1555 and XRGB1555). | 
|  | * | 
|  | * For ARGB, set the alpha value to 0, and enable alpha-blending when | 
|  | * the A bit is 0. This maps A=0 to alpha=0 and A=1 to alpha=255. | 
|  | * | 
|  | * For XRGB, set the alpha value to the plane-wide alpha value and | 
|  | * enable alpha-blending regardless of the X bit value. | 
|  | */ | 
|  | if (plane->format->fourcc != DRM_FORMAT_XRGB1555) | 
|  | rcar_du_plane_write(rcdu, index, PnALPHAR, PnALPHAR_ABIT_0); | 
|  | else | 
|  | rcar_du_plane_write(rcdu, index, PnALPHAR, | 
|  | PnALPHAR_ABIT_X | plane->alpha); | 
|  |  | 
|  | pnmr = PnMR_BM_MD | plane->format->pnmr; | 
|  |  | 
|  | /* Disable color keying when requested. YUV formats have the | 
|  | * PnMR_SPIM_TP_OFF bit set in their pnmr field, disabling color keying | 
|  | * automatically. | 
|  | */ | 
|  | if ((plane->colorkey & RCAR_DU_COLORKEY_MASK) == RCAR_DU_COLORKEY_NONE) | 
|  | pnmr |= PnMR_SPIM_TP_OFF; | 
|  |  | 
|  | /* For packed YUV formats we need to select the U/V order. */ | 
|  | if (plane->format->fourcc == DRM_FORMAT_YUYV) | 
|  | pnmr |= PnMR_YCDF_YUYV; | 
|  |  | 
|  | rcar_du_plane_write(rcdu, index, PnMR, pnmr); | 
|  |  | 
|  | switch (plane->format->fourcc) { | 
|  | case DRM_FORMAT_RGB565: | 
|  | colorkey = ((plane->colorkey & 0xf80000) >> 8) | 
|  | | ((plane->colorkey & 0x00fc00) >> 5) | 
|  | | ((plane->colorkey & 0x0000f8) >> 3); | 
|  | rcar_du_plane_write(rcdu, index, PnTC2R, colorkey); | 
|  | break; | 
|  |  | 
|  | case DRM_FORMAT_ARGB1555: | 
|  | case DRM_FORMAT_XRGB1555: | 
|  | colorkey = ((plane->colorkey & 0xf80000) >> 9) | 
|  | | ((plane->colorkey & 0x00f800) >> 6) | 
|  | | ((plane->colorkey & 0x0000f8) >> 3); | 
|  | rcar_du_plane_write(rcdu, index, PnTC2R, colorkey); | 
|  | break; | 
|  |  | 
|  | case DRM_FORMAT_XRGB8888: | 
|  | case DRM_FORMAT_ARGB8888: | 
|  | rcar_du_plane_write(rcdu, index, PnTC3R, | 
|  | PnTC3R_CODE | (plane->colorkey & 0xffffff)); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | static void __rcar_du_plane_setup(struct rcar_du_plane *plane, | 
|  | unsigned int index) | 
|  | { | 
|  | struct rcar_du_device *rcdu = plane->dev; | 
|  | u32 ddcr2 = PnDDCR2_CODE; | 
|  | u32 ddcr4; | 
|  | u32 mwr; | 
|  |  | 
|  | /* Data format | 
|  | * | 
|  | * The data format is selected by the DDDF field in PnMR and the EDF | 
|  | * field in DDCR4. | 
|  | */ | 
|  | ddcr4 = rcar_du_plane_read(rcdu, index, PnDDCR4); | 
|  | ddcr4 &= ~PnDDCR4_EDF_MASK; | 
|  | ddcr4 |= plane->format->edf | PnDDCR4_CODE; | 
|  |  | 
|  | rcar_du_plane_setup_mode(plane, index); | 
|  |  | 
|  | if (plane->format->planes == 2) { | 
|  | if (plane->hwindex != index) { | 
|  | if (plane->format->fourcc == DRM_FORMAT_NV12 || | 
|  | plane->format->fourcc == DRM_FORMAT_NV21) | 
|  | ddcr2 |= PnDDCR2_Y420; | 
|  |  | 
|  | if (plane->format->fourcc == DRM_FORMAT_NV21) | 
|  | ddcr2 |= PnDDCR2_NV21; | 
|  |  | 
|  | ddcr2 |= PnDDCR2_DIVU; | 
|  | } else { | 
|  | ddcr2 |= PnDDCR2_DIVY; | 
|  | } | 
|  | } | 
|  |  | 
|  | rcar_du_plane_write(rcdu, index, PnDDCR2, ddcr2); | 
|  | rcar_du_plane_write(rcdu, index, PnDDCR4, ddcr4); | 
|  |  | 
|  | /* Memory pitch (expressed in pixels) */ | 
|  | if (plane->format->planes == 2) | 
|  | mwr = plane->pitch; | 
|  | else | 
|  | mwr = plane->pitch * 8 / plane->format->bpp; | 
|  |  | 
|  | rcar_du_plane_write(rcdu, index, PnMWR, mwr); | 
|  |  | 
|  | /* Destination position and size */ | 
|  | rcar_du_plane_write(rcdu, index, PnDSXR, plane->width); | 
|  | rcar_du_plane_write(rcdu, index, PnDSYR, plane->height); | 
|  | rcar_du_plane_write(rcdu, index, PnDPXR, plane->dst_x); | 
|  | rcar_du_plane_write(rcdu, index, PnDPYR, plane->dst_y); | 
|  |  | 
|  | /* Wrap-around and blinking, disabled */ | 
|  | rcar_du_plane_write(rcdu, index, PnWASPR, 0); | 
|  | rcar_du_plane_write(rcdu, index, PnWAMWR, 4095); | 
|  | rcar_du_plane_write(rcdu, index, PnBTR, 0); | 
|  | rcar_du_plane_write(rcdu, index, PnMLR, 0); | 
|  | } | 
|  |  | 
|  | void rcar_du_plane_setup(struct rcar_du_plane *plane) | 
|  | { | 
|  | __rcar_du_plane_setup(plane, plane->hwindex); | 
|  | if (plane->format->planes == 2) | 
|  | __rcar_du_plane_setup(plane, (plane->hwindex + 1) % 8); | 
|  |  | 
|  | rcar_du_plane_update_base(plane); | 
|  | } | 
|  |  | 
|  | static int | 
|  | rcar_du_plane_update(struct drm_plane *plane, struct drm_crtc *crtc, | 
|  | struct drm_framebuffer *fb, int crtc_x, int crtc_y, | 
|  | unsigned int crtc_w, unsigned int crtc_h, | 
|  | uint32_t src_x, uint32_t src_y, | 
|  | uint32_t src_w, uint32_t src_h) | 
|  | { | 
|  | struct rcar_du_plane *rplane = to_rcar_plane(plane); | 
|  | struct rcar_du_device *rcdu = plane->dev->dev_private; | 
|  | const struct rcar_du_format_info *format; | 
|  | unsigned int nplanes; | 
|  | int ret; | 
|  |  | 
|  | format = rcar_du_format_info(fb->pixel_format); | 
|  | if (format == NULL) { | 
|  | dev_dbg(rcdu->dev, "%s: unsupported format %08x\n", __func__, | 
|  | fb->pixel_format); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | if (src_w >> 16 != crtc_w || src_h >> 16 != crtc_h) { | 
|  | dev_dbg(rcdu->dev, "%s: scaling not supported\n", __func__); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | nplanes = rplane->format ? rplane->format->planes : 0; | 
|  |  | 
|  | /* Reallocate hardware planes if the number of required planes has | 
|  | * changed. | 
|  | */ | 
|  | if (format->planes != nplanes) { | 
|  | rcar_du_plane_release(rplane); | 
|  | ret = rcar_du_plane_reserve(rplane, format); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | rplane->crtc = crtc; | 
|  | rplane->format = format; | 
|  | rplane->pitch = fb->pitches[0]; | 
|  |  | 
|  | rplane->src_x = src_x >> 16; | 
|  | rplane->src_y = src_y >> 16; | 
|  | rplane->dst_x = crtc_x; | 
|  | rplane->dst_y = crtc_y; | 
|  | rplane->width = crtc_w; | 
|  | rplane->height = crtc_h; | 
|  |  | 
|  | rcar_du_plane_compute_base(rplane, fb); | 
|  | rcar_du_plane_setup(rplane); | 
|  |  | 
|  | mutex_lock(&rcdu->planes.lock); | 
|  | rplane->enabled = true; | 
|  | rcar_du_crtc_update_planes(rplane->crtc); | 
|  | mutex_unlock(&rcdu->planes.lock); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int rcar_du_plane_disable(struct drm_plane *plane) | 
|  | { | 
|  | struct rcar_du_device *rcdu = plane->dev->dev_private; | 
|  | struct rcar_du_plane *rplane = to_rcar_plane(plane); | 
|  |  | 
|  | if (!rplane->enabled) | 
|  | return 0; | 
|  |  | 
|  | mutex_lock(&rcdu->planes.lock); | 
|  | rplane->enabled = false; | 
|  | rcar_du_crtc_update_planes(rplane->crtc); | 
|  | mutex_unlock(&rcdu->planes.lock); | 
|  |  | 
|  | rcar_du_plane_release(rplane); | 
|  |  | 
|  | rplane->crtc = NULL; | 
|  | rplane->format = NULL; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Both the .set_property and the .update_plane operations are called with the | 
|  | * mode_config lock held. There is this no need to explicitly protect access to | 
|  | * the alpha and colorkey fields and the mode register. | 
|  | */ | 
|  | static void rcar_du_plane_set_alpha(struct rcar_du_plane *plane, u32 alpha) | 
|  | { | 
|  | if (plane->alpha == alpha) | 
|  | return; | 
|  |  | 
|  | plane->alpha = alpha; | 
|  | if (!plane->enabled || plane->format->fourcc != DRM_FORMAT_XRGB1555) | 
|  | return; | 
|  |  | 
|  | rcar_du_plane_setup_mode(plane, plane->hwindex); | 
|  | } | 
|  |  | 
|  | static void rcar_du_plane_set_colorkey(struct rcar_du_plane *plane, | 
|  | u32 colorkey) | 
|  | { | 
|  | if (plane->colorkey == colorkey) | 
|  | return; | 
|  |  | 
|  | plane->colorkey = colorkey; | 
|  | if (!plane->enabled) | 
|  | return; | 
|  |  | 
|  | rcar_du_plane_setup_mode(plane, plane->hwindex); | 
|  | } | 
|  |  | 
|  | static void rcar_du_plane_set_zpos(struct rcar_du_plane *plane, | 
|  | unsigned int zpos) | 
|  | { | 
|  | struct rcar_du_device *rcdu = plane->dev; | 
|  |  | 
|  | mutex_lock(&rcdu->planes.lock); | 
|  | if (plane->zpos == zpos) | 
|  | goto done; | 
|  |  | 
|  | plane->zpos = zpos; | 
|  | if (!plane->enabled) | 
|  | goto done; | 
|  |  | 
|  | rcar_du_crtc_update_planes(plane->crtc); | 
|  |  | 
|  | done: | 
|  | mutex_unlock(&rcdu->planes.lock); | 
|  | } | 
|  |  | 
|  | static int rcar_du_plane_set_property(struct drm_plane *plane, | 
|  | struct drm_property *property, | 
|  | uint64_t value) | 
|  | { | 
|  | struct rcar_du_device *rcdu = plane->dev->dev_private; | 
|  | struct rcar_du_plane *rplane = to_rcar_plane(plane); | 
|  |  | 
|  | if (property == rcdu->planes.alpha) | 
|  | rcar_du_plane_set_alpha(rplane, value); | 
|  | else if (property == rcdu->planes.colorkey) | 
|  | rcar_du_plane_set_colorkey(rplane, value); | 
|  | else if (property == rcdu->planes.zpos) | 
|  | rcar_du_plane_set_zpos(rplane, value); | 
|  | else | 
|  | return -EINVAL; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct drm_plane_funcs rcar_du_plane_funcs = { | 
|  | .update_plane = rcar_du_plane_update, | 
|  | .disable_plane = rcar_du_plane_disable, | 
|  | .set_property = rcar_du_plane_set_property, | 
|  | .destroy = drm_plane_cleanup, | 
|  | }; | 
|  |  | 
|  | static const uint32_t formats[] = { | 
|  | DRM_FORMAT_RGB565, | 
|  | DRM_FORMAT_ARGB1555, | 
|  | DRM_FORMAT_XRGB1555, | 
|  | DRM_FORMAT_XRGB8888, | 
|  | DRM_FORMAT_ARGB8888, | 
|  | DRM_FORMAT_UYVY, | 
|  | DRM_FORMAT_YUYV, | 
|  | DRM_FORMAT_NV12, | 
|  | DRM_FORMAT_NV21, | 
|  | DRM_FORMAT_NV16, | 
|  | }; | 
|  |  | 
|  | int rcar_du_plane_init(struct rcar_du_device *rcdu) | 
|  | { | 
|  | unsigned int i; | 
|  |  | 
|  | mutex_init(&rcdu->planes.lock); | 
|  | rcdu->planes.free = 0xff; | 
|  |  | 
|  | rcdu->planes.alpha = | 
|  | drm_property_create_range(rcdu->ddev, 0, "alpha", 0, 255); | 
|  | if (rcdu->planes.alpha == NULL) | 
|  | return -ENOMEM; | 
|  |  | 
|  | /* The color key is expressed as an RGB888 triplet stored in a 32-bit | 
|  | * integer in XRGB8888 format. Bit 24 is used as a flag to disable (0) | 
|  | * or enable source color keying (1). | 
|  | */ | 
|  | rcdu->planes.colorkey = | 
|  | drm_property_create_range(rcdu->ddev, 0, "colorkey", | 
|  | 0, 0x01ffffff); | 
|  | if (rcdu->planes.colorkey == NULL) | 
|  | return -ENOMEM; | 
|  |  | 
|  | rcdu->planes.zpos = | 
|  | drm_property_create_range(rcdu->ddev, 0, "zpos", 1, 7); | 
|  | if (rcdu->planes.zpos == NULL) | 
|  | return -ENOMEM; | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(rcdu->planes.planes); ++i) { | 
|  | struct rcar_du_plane *plane = &rcdu->planes.planes[i]; | 
|  |  | 
|  | plane->dev = rcdu; | 
|  | plane->hwindex = -1; | 
|  | plane->alpha = 255; | 
|  | plane->colorkey = RCAR_DU_COLORKEY_NONE; | 
|  | plane->zpos = 0; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int rcar_du_plane_register(struct rcar_du_device *rcdu) | 
|  | { | 
|  | unsigned int i; | 
|  | int ret; | 
|  |  | 
|  | for (i = 0; i < RCAR_DU_NUM_KMS_PLANES; ++i) { | 
|  | struct rcar_du_kms_plane *plane; | 
|  |  | 
|  | plane = devm_kzalloc(rcdu->dev, sizeof(*plane), GFP_KERNEL); | 
|  | if (plane == NULL) | 
|  | return -ENOMEM; | 
|  |  | 
|  | plane->hwplane = &rcdu->planes.planes[i + 2]; | 
|  | plane->hwplane->zpos = 1; | 
|  |  | 
|  | ret = drm_plane_init(rcdu->ddev, &plane->plane, | 
|  | (1 << rcdu->num_crtcs) - 1, | 
|  | &rcar_du_plane_funcs, formats, | 
|  | ARRAY_SIZE(formats), false); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | drm_object_attach_property(&plane->plane.base, | 
|  | rcdu->planes.alpha, 255); | 
|  | drm_object_attach_property(&plane->plane.base, | 
|  | rcdu->planes.colorkey, | 
|  | RCAR_DU_COLORKEY_NONE); | 
|  | drm_object_attach_property(&plane->plane.base, | 
|  | rcdu->planes.zpos, 1); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } |