Path: blob/master/drivers/gpu/drm/arm/display/komeda/komeda_framebuffer.c
26519 views
// SPDX-License-Identifier: GPL-2.01/*2* (C) COPYRIGHT 2018 ARM Limited. All rights reserved.3* Author: James.Qian.Wang <[email protected]>4*5*/6#include <drm/drm_device.h>7#include <drm/drm_fb_dma_helper.h>8#include <drm/drm_gem.h>9#include <drm/drm_gem_dma_helper.h>10#include <drm/drm_gem_framebuffer_helper.h>1112#include "komeda_framebuffer.h"13#include "komeda_dev.h"1415static void komeda_fb_destroy(struct drm_framebuffer *fb)16{17struct komeda_fb *kfb = to_kfb(fb);18u32 i;1920for (i = 0; i < fb->format->num_planes; i++)21drm_gem_object_put(fb->obj[i]);2223drm_framebuffer_cleanup(fb);24kfree(kfb);25}2627static int komeda_fb_create_handle(struct drm_framebuffer *fb,28struct drm_file *file, u32 *handle)29{30return drm_gem_handle_create(file, fb->obj[0], handle);31}3233static const struct drm_framebuffer_funcs komeda_fb_funcs = {34.destroy = komeda_fb_destroy,35.create_handle = komeda_fb_create_handle,36};3738static int39komeda_fb_afbc_size_check(struct komeda_fb *kfb, struct drm_file *file,40const struct drm_mode_fb_cmd2 *mode_cmd)41{42struct drm_framebuffer *fb = &kfb->base;43const struct drm_format_info *info = fb->format;44struct drm_gem_object *obj;45u32 alignment_w = 0, alignment_h = 0, alignment_header, n_blocks, bpp;46u64 min_size;4748obj = drm_gem_object_lookup(file, mode_cmd->handles[0]);49if (!obj) {50DRM_DEBUG_KMS("Failed to lookup GEM object\n");51return -ENOENT;52}5354switch (fb->modifier & AFBC_FORMAT_MOD_BLOCK_SIZE_MASK) {55case AFBC_FORMAT_MOD_BLOCK_SIZE_32x8:56alignment_w = 32;57alignment_h = 8;58break;59case AFBC_FORMAT_MOD_BLOCK_SIZE_16x16:60alignment_w = 16;61alignment_h = 16;62break;63default:64WARN(1, "Invalid AFBC_FORMAT_MOD_BLOCK_SIZE: %lld.\n",65fb->modifier & AFBC_FORMAT_MOD_BLOCK_SIZE_MASK);66break;67}6869/* tiled header afbc */70if (fb->modifier & AFBC_FORMAT_MOD_TILED) {71alignment_w *= AFBC_TH_LAYOUT_ALIGNMENT;72alignment_h *= AFBC_TH_LAYOUT_ALIGNMENT;73alignment_header = AFBC_TH_BODY_START_ALIGNMENT;74} else {75alignment_header = AFBC_BODY_START_ALIGNMENT;76}7778kfb->aligned_w = ALIGN(fb->width, alignment_w);79kfb->aligned_h = ALIGN(fb->height, alignment_h);8081if (fb->offsets[0] % alignment_header) {82DRM_DEBUG_KMS("afbc offset alignment check failed.\n");83goto check_failed;84}8586n_blocks = (kfb->aligned_w * kfb->aligned_h) / AFBC_SUPERBLK_PIXELS;87kfb->offset_payload = ALIGN(n_blocks * AFBC_HEADER_SIZE,88alignment_header);8990bpp = komeda_get_afbc_format_bpp(info, fb->modifier);91kfb->afbc_size = kfb->offset_payload + n_blocks *92ALIGN(bpp * AFBC_SUPERBLK_PIXELS / 8,93AFBC_SUPERBLK_ALIGNMENT);94min_size = kfb->afbc_size + fb->offsets[0];95if (min_size > obj->size) {96DRM_DEBUG_KMS("afbc size check failed, obj_size: 0x%zx. min_size 0x%llx.\n",97obj->size, min_size);98goto check_failed;99}100101fb->obj[0] = obj;102return 0;103104check_failed:105drm_gem_object_put(obj);106return -EINVAL;107}108109static int110komeda_fb_none_afbc_size_check(struct komeda_dev *mdev, struct komeda_fb *kfb,111struct drm_file *file,112const struct drm_mode_fb_cmd2 *mode_cmd)113{114struct drm_framebuffer *fb = &kfb->base;115const struct drm_format_info *info = fb->format;116struct drm_gem_object *obj;117u32 i, block_h;118u64 min_size;119120if (komeda_fb_check_src_coords(kfb, 0, 0, fb->width, fb->height))121return -EINVAL;122123for (i = 0; i < info->num_planes; i++) {124obj = drm_gem_object_lookup(file, mode_cmd->handles[i]);125if (!obj) {126DRM_DEBUG_KMS("Failed to lookup GEM object\n");127return -ENOENT;128}129fb->obj[i] = obj;130131block_h = drm_format_info_block_height(info, i);132if ((fb->pitches[i] * block_h) % mdev->chip.bus_width) {133DRM_DEBUG_KMS("Pitch[%d]: 0x%x doesn't align to 0x%x\n",134i, fb->pitches[i], mdev->chip.bus_width);135return -EINVAL;136}137138min_size = komeda_fb_get_pixel_addr(kfb, 0, fb->height, i)139- to_drm_gem_dma_obj(obj)->dma_addr;140if (obj->size < min_size) {141DRM_DEBUG_KMS("The fb->obj[%d] size: 0x%zx lower than the minimum requirement: 0x%llx.\n",142i, obj->size, min_size);143return -EINVAL;144}145}146147if (fb->format->num_planes == 3) {148if (fb->pitches[1] != fb->pitches[2]) {149DRM_DEBUG_KMS("The pitch[1] and [2] are not same\n");150return -EINVAL;151}152}153154return 0;155}156157struct drm_framebuffer *158komeda_fb_create(struct drm_device *dev, struct drm_file *file,159const struct drm_format_info *info,160const struct drm_mode_fb_cmd2 *mode_cmd)161{162struct komeda_dev *mdev = dev->dev_private;163struct komeda_fb *kfb;164int ret = 0, i;165166kfb = kzalloc(sizeof(*kfb), GFP_KERNEL);167if (!kfb)168return ERR_PTR(-ENOMEM);169170kfb->format_caps = komeda_get_format_caps(&mdev->fmt_tbl,171mode_cmd->pixel_format,172mode_cmd->modifier[0]);173if (!kfb->format_caps) {174DRM_DEBUG_KMS("FMT %x is not supported.\n",175mode_cmd->pixel_format);176kfree(kfb);177return ERR_PTR(-EINVAL);178}179180drm_helper_mode_fill_fb_struct(dev, &kfb->base, info, mode_cmd);181182if (kfb->base.modifier)183ret = komeda_fb_afbc_size_check(kfb, file, mode_cmd);184else185ret = komeda_fb_none_afbc_size_check(mdev, kfb, file, mode_cmd);186if (ret < 0)187goto err_cleanup;188189ret = drm_framebuffer_init(dev, &kfb->base, &komeda_fb_funcs);190if (ret < 0) {191DRM_DEBUG_KMS("failed to initialize fb\n");192193goto err_cleanup;194}195196kfb->is_va = mdev->iommu ? true : false;197198return &kfb->base;199200err_cleanup:201for (i = 0; i < kfb->base.format->num_planes; i++)202drm_gem_object_put(kfb->base.obj[i]);203204kfree(kfb);205return ERR_PTR(ret);206}207208int komeda_fb_check_src_coords(const struct komeda_fb *kfb,209u32 src_x, u32 src_y, u32 src_w, u32 src_h)210{211const struct drm_framebuffer *fb = &kfb->base;212const struct drm_format_info *info = fb->format;213u32 block_w = drm_format_info_block_width(fb->format, 0);214u32 block_h = drm_format_info_block_height(fb->format, 0);215216if ((src_x + src_w > fb->width) || (src_y + src_h > fb->height)) {217DRM_DEBUG_ATOMIC("Invalid source coordinate.\n");218return -EINVAL;219}220221if ((src_x % info->hsub) || (src_w % info->hsub) ||222(src_y % info->vsub) || (src_h % info->vsub)) {223DRM_DEBUG_ATOMIC("Wrong subsampling dimension x:%d, y:%d, w:%d, h:%d for format: %x.\n",224src_x, src_y, src_w, src_h, info->format);225return -EINVAL;226}227228if ((src_x % block_w) || (src_w % block_w) ||229(src_y % block_h) || (src_h % block_h)) {230DRM_DEBUG_ATOMIC("x:%d, y:%d, w:%d, h:%d should be multiple of block_w/h for format: %x.\n",231src_x, src_y, src_w, src_h, info->format);232return -EINVAL;233}234235return 0;236}237238dma_addr_t239komeda_fb_get_pixel_addr(struct komeda_fb *kfb, int x, int y, int plane)240{241struct drm_framebuffer *fb = &kfb->base;242const struct drm_gem_dma_object *obj;243u32 offset, plane_x, plane_y, block_w, block_sz;244245if (plane >= fb->format->num_planes) {246DRM_DEBUG_KMS("Out of max plane num.\n");247return -EINVAL;248}249250obj = drm_fb_dma_get_gem_obj(fb, plane);251252offset = fb->offsets[plane];253if (!fb->modifier) {254block_w = drm_format_info_block_width(fb->format, plane);255block_sz = fb->format->char_per_block[plane];256plane_x = x / (plane ? fb->format->hsub : 1);257plane_y = y / (plane ? fb->format->vsub : 1);258259offset += (plane_x / block_w) * block_sz260+ plane_y * fb->pitches[plane];261}262263return obj->dma_addr + offset;264}265266/* if the fb can be supported by a specific layer */267bool komeda_fb_is_layer_supported(struct komeda_fb *kfb, u32 layer_type,268u32 rot)269{270struct drm_framebuffer *fb = &kfb->base;271struct komeda_dev *mdev = fb->dev->dev_private;272u32 fourcc = fb->format->format;273u64 modifier = fb->modifier;274bool supported;275276supported = komeda_format_mod_supported(&mdev->fmt_tbl, layer_type,277fourcc, modifier, rot);278if (!supported)279DRM_DEBUG_ATOMIC("Layer TYPE: %d doesn't support fb FMT: %p4cc with modifier: 0x%llx.\n",280layer_type, &fourcc, modifier);281282return supported;283}284285286