From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id E2DFAEB64DC for ; Fri, 21 Jul 2023 09:06:47 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231319AbjGUJGq convert rfc822-to-8bit (ORCPT ); Fri, 21 Jul 2023 05:06:46 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60788 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S231305AbjGUJGp (ORCPT ); Fri, 21 Jul 2023 05:06:45 -0400 Received: from fd01.gateway.ufhost.com (fd01.gateway.ufhost.com [61.152.239.71]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 1528D2D7D; Fri, 21 Jul 2023 02:06:41 -0700 (PDT) Received: from EXMBX165.cuchost.com (unknown [175.102.18.54]) (using TLSv1 with cipher DHE-RSA-AES256-SHA (256/256 bits)) (Client CN "EXMBX165", Issuer "EXMBX165" (not verified)) by fd01.gateway.ufhost.com (Postfix) with ESMTP id 829F424DED7; Fri, 21 Jul 2023 17:06:33 +0800 (CST) Received: from EXMBX061.cuchost.com (172.16.6.61) by EXMBX165.cuchost.com (172.16.6.75) with Microsoft SMTP Server (TLS) id 15.0.1497.42; Fri, 21 Jul 2023 17:06:33 +0800 Received: from [192.168.60.102] (180.164.60.184) by EXMBX061.cuchost.com (172.16.6.61) with Microsoft SMTP Server (TLS) id 15.0.1497.42; Fri, 21 Jul 2023 17:06:32 +0800 Message-ID: Date: Fri, 21 Jul 2023 17:06:32 +0800 MIME-Version: 1.0 User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:102.0) Gecko/20100101 Thunderbird/102.13.0 Subject: Re: [PATCH 5/9] drm/verisilicon: Add mode config funcs Content-Language: en-US To: Thomas Zimmermann , , , , , , CC: David Airlie , Daniel Vetter , "Rob Herring" , Krzysztof Kozlowski , Conor Dooley , "Emil Renner Berthing" , Paul Walmsley , Palmer Dabbelt , Albert Ou , Maarten Lankhorst , Maxime Ripard , Philipp Zabel , Sumit Semwal , , Bjorn Andersson , Heiko Stuebner , Shawn Guo , Jagan Teki , Chris Morgan , Jack Zhu , "Shengyang Chen" , Changhuang Liang References: <20230602074043.33872-1-keith.zhao@starfivetech.com> <20230602074043.33872-6-keith.zhao@starfivetech.com> <30008535-8606-fc6d-9c07-23d46f59c957@suse.de> From: Keith Zhao In-Reply-To: <30008535-8606-fc6d-9c07-23d46f59c957@suse.de> Content-Type: text/plain; charset="UTF-8" X-Originating-IP: [180.164.60.184] X-ClientProxiedBy: EXCAS064.cuchost.com (172.16.6.24) To EXMBX061.cuchost.com (172.16.6.61) X-YovoleRuleAgent: yovoleflag Content-Transfer-Encoding: 8BIT Precedence: bulk List-ID: X-Mailing-List: linux-media@vger.kernel.org On 2023/6/21 19:04, Thomas Zimmermann wrote: > Hi Keith > > Am 02.06.23 um 09:40 schrieb Keith Zhao: >> Add mode setting functions for JH7110 SoC. >> >> Signed-off-by: Keith Zhao >> --- >>   drivers/gpu/drm/verisilicon/Makefile |   1 + >>   drivers/gpu/drm/verisilicon/vs_drv.c |   3 + > >>   drivers/gpu/drm/verisilicon/vs_fb.c  | 181 +++++++++++++++++++++++++++ >>   drivers/gpu/drm/verisilicon/vs_fb.h  |  15 +++ > > I'd call these files vs_modeset.{c,h} to be consistent with the rest of the drivers. > >>   4 files changed, 200 insertions(+) >>   create mode 100644 drivers/gpu/drm/verisilicon/vs_fb.c >>   create mode 100644 drivers/gpu/drm/verisilicon/vs_fb.h >> >> diff --git a/drivers/gpu/drm/verisilicon/Makefile b/drivers/gpu/drm/verisilicon/Makefile >> index 30360e370e47..38254dc5d98d 100644 >> --- a/drivers/gpu/drm/verisilicon/Makefile >> +++ b/drivers/gpu/drm/verisilicon/Makefile >> @@ -1,6 +1,7 @@ >>   # SPDX-License-Identifier: GPL-2.0 >>     vs_drm-objs := vs_drv.o \ >> +        vs_fb.o \ >>           vs_gem.o >>     obj-$(CONFIG_DRM_VERISILICON) += vs_drm.o >> diff --git a/drivers/gpu/drm/verisilicon/vs_drv.c b/drivers/gpu/drm/verisilicon/vs_drv.c >> index e0a2fc43b55f..d84aacd751bc 100644 >> --- a/drivers/gpu/drm/verisilicon/vs_drv.c >> +++ b/drivers/gpu/drm/verisilicon/vs_drv.c >> @@ -30,6 +30,7 @@ >>   #include >>     #include "vs_drv.h" >> +#include "vs_fb.h" >>   #include "vs_gem.h" >>     #define DRV_NAME    "starfive" >> @@ -118,6 +119,8 @@ static int vs_drm_bind(struct device *dev) >>       if (ret) >>           goto err_mode; >>   +    vs_mode_config_init(drm_dev); >> + >>       ret = drm_vblank_init(drm_dev, drm_dev->mode_config.num_crtc); >>       if (ret) >>           goto err_bind; >> diff --git a/drivers/gpu/drm/verisilicon/vs_fb.c b/drivers/gpu/drm/verisilicon/vs_fb.c >> new file mode 100644 >> index 000000000000..3e85f7365084 >> --- /dev/null >> +++ b/drivers/gpu/drm/verisilicon/vs_fb.c >> @@ -0,0 +1,181 @@ >> +// SPDX-License-Identifier: GPL-2.0 >> +/* >> + * Copyright (C) 2023 VeriSilicon Holdings Co., Ltd. >> + */ >> + >> +#include >> +#include >> + >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> +#include >> + >> +#include "vs_fb.h" >> +#include "vs_gem.h" >> + >> +#define fourcc_mod_vs_get_type(val) \ >> +    (((val) & DRM_FORMAT_MOD_VS_TYPE_MASK) >> 54) >> + >> +static struct drm_framebuffer_funcs vs_fb_funcs = { >> +    .create_handle    = drm_gem_fb_create_handle, >> +    .destroy    = drm_gem_fb_destroy, >> +    .dirty        = drm_atomic_helper_dirtyfb, >> +}; >> + >> +static struct drm_framebuffer * >> +vs_fb_alloc(struct drm_device *dev, const struct drm_mode_fb_cmd2 *mode_cmd, >> +        struct vs_gem_object **obj, unsigned int num_planes) >> +{ >> +    struct drm_framebuffer *fb; >> +    int ret, i; >> + >> +    fb = kzalloc(sizeof(*fb), GFP_KERNEL); >> +    if (!fb) >> +        return ERR_PTR(-ENOMEM); >> + >> +    drm_helper_mode_fill_fb_struct(dev, fb, mode_cmd); >> + >> +    for (i = 0; i < num_planes; i++) >> +        fb->obj[i] = &obj[i]->base; >> + >> +    ret = drm_framebuffer_init(dev, fb, &vs_fb_funcs); >> +    if (ret) { >> +        dev_err(dev->dev, "Failed to initialize framebuffer: %d\n", >> +            ret); >> +        kfree(fb); >> +        return ERR_PTR(ret); >> +    } >> + >> +    return fb; >> +} >> + >> +static struct drm_framebuffer *vs_fb_create(struct drm_device *dev, >> +                        struct drm_file *file_priv, >> +                        const struct drm_mode_fb_cmd2 *mode_cmd) >> +{ >> +    struct drm_framebuffer *fb; >> +    const struct drm_format_info *info; >> +    struct vs_gem_object *objs[MAX_NUM_PLANES]; >> +    struct drm_gem_object *obj; >> +    unsigned int height, size; >> +    unsigned char i, num_planes; >> +    int ret = 0; >> + >> +    info = drm_get_format_info(dev, mode_cmd); >> +    if (!info) >> +        return ERR_PTR(-EINVAL); >> + >> +    num_planes = info->num_planes; >> +    if (num_planes > MAX_NUM_PLANES) >> +        return ERR_PTR(-EINVAL); >> + >> +    for (i = 0; i < num_planes; i++) { >> +        obj = drm_gem_object_lookup(file_priv, mode_cmd->handles[i]); >> +        if (!obj) { >> +            dev_err(dev->dev, "Failed to lookup GEM object.\n"); >> +            ret = -ENXIO; >> +            goto err; >> +        } >> + >> +        height = drm_format_info_plane_height(info, >> +                              mode_cmd->height, i); >> + >> +        size = height * mode_cmd->pitches[i] + mode_cmd->offsets[i]; >> + >> +        if (obj->size < size) { >> +            drm_gem_object_put(obj); >> + >> +            ret = -EINVAL; >> +            goto err; >> +        } >> + >> +        objs[i] = to_vs_gem_object(obj); >> +    } >> + >> +    fb = vs_fb_alloc(dev, mode_cmd, objs, i); >> +    if (IS_ERR(fb)) { >> +        ret = PTR_ERR(fb); >> +        goto err; >> +    } >> + >> +    return fb; >> + >> +err: >> +    for (; i > 0; i--) >> +        drm_gem_object_put(&objs[i - 1]->base); >> + >> +    return ERR_PTR(ret); >> +} >> + >> +struct vs_gem_object *vs_fb_get_gem_obj(struct drm_framebuffer *fb, >> +                    unsigned char plane) >> +{ >> +    if (plane > MAX_NUM_PLANES) >> +        return NULL; >> + >> +    return to_vs_gem_object(fb->obj[plane]); >> +} >> + >> +static const struct drm_format_info vs_formats[] = { >> +    {.format = DRM_FORMAT_NV12, .depth = 0, .num_planes = 2, .char_per_block = { 20, 40, 0 }, >> +     .block_w = { 4, 4, 0 }, .block_h = { 4, 4, 0 }, .hsub = 2, .vsub = 2, .is_yuv = true}, >> +    {.format = DRM_FORMAT_YUV444, .depth = 0, .num_planes = 3, .char_per_block = { 20, 20, 20 }, >> +     .block_w = { 4, 4, 4 }, .block_h = { 4, 4, 4 }, .hsub = 1, .vsub = 1, .is_yuv = true}, >> +}; >> + >> +static const struct drm_format_info * >> +vs_lookup_format_info(const struct drm_format_info formats[], >> +              int num_formats, u32 format) >> +{ >> +    int i; >> + >> +    for (i = 0; i < num_formats; i++) { >> +        if (formats[i].format == format) >> +            return &formats[i]; >> +    } >> + >> +    return NULL; >> +} >> + >> +static const struct drm_format_info * >> +vs_get_format_info(const struct drm_mode_fb_cmd2 *cmd) >> +{ >> +    if (fourcc_mod_vs_get_type(cmd->modifier[0]) == >> +        DRM_FORMAT_MOD_VS_TYPE_CUSTOM_10BIT) >> +        return vs_lookup_format_info(vs_formats, ARRAY_SIZE(vs_formats), >> +                                     cmd->pixel_format); >> +    else >> +        return NULL; >> +} >> + >> +static const struct drm_mode_config_funcs vs_mode_config_funcs = { >> +    .fb_create             = vs_fb_create, > > Maybe I'm missing something here, but it looks like you can call > drm_gem_fb_create_with_funcs() to create the framebuffer. > That's a brilliant suggestion!!! >> +    .get_format_info     = vs_get_format_info, >> +    .output_poll_changed = drm_fb_helper_output_poll_changed, >> +    .atomic_check         = drm_atomic_helper_check, >> +    .atomic_commit         = drm_atomic_helper_commit, >> +}; >> + >> +static struct drm_mode_config_helper_funcs vs_mode_config_helpers = { >> +    .atomic_commit_tail = drm_atomic_helper_commit_tail_rpm, >> +}; >> + >> +void vs_mode_config_init(struct drm_device *dev) >> +{ > > If possible, move the call to drm_mode_config_init() into this function. > move drm_mode_config_init() into vs_mode_config_init. and vs_mode_config_init must be called ahead component_bind_all like this: static int vs_drm_bind(struct device *dev) { ...... vs_mode_config_init(drm_dev); /* Now try and bind all our sub-components */ ret = component_bind_all(dev, drm_dev); if (ret) goto err_mode; ...... } >> +    dev->mode_config.fb_modifiers_not_supported = false; >> + >> +    if (dev->mode_config.max_width == 0 || >> +        dev->mode_config.max_height == 0) { >> +        dev->mode_config.min_width  = 0; >> +        dev->mode_config.min_height = 0; >> +        dev->mode_config.max_width  = 4096; >> +        dev->mode_config.max_height = 4096; >> +    } >> +    dev->mode_config.funcs = &vs_mode_config_funcs; >> +    dev->mode_config.helper_private = &vs_mode_config_helpers; >> +} >> diff --git a/drivers/gpu/drm/verisilicon/vs_fb.h b/drivers/gpu/drm/verisilicon/vs_fb.h >> new file mode 100644 >> index 000000000000..78dda8e42894 >> --- /dev/null >> +++ b/drivers/gpu/drm/verisilicon/vs_fb.h >> @@ -0,0 +1,15 @@ >> +/* SPDX-License-Identifier: GPL-2.0 */ >> +/* >> + * Copyright (C) 2023 VeriSilicon Holdings Co., Ltd. >> + */ >> + >> +#ifndef __VS_FB_H__ >> +#define __VS_FB_H__ >> + >> +#define MAX_NUM_PLANES        3 /* colour format plane */ > > There's DRM_FORMAT_MAX_PLANES already. Please don't introduce a constant with the same purpose. > ok good idea! > Best regards > Thomas > >> + >> +struct vs_gem_object *vs_fb_get_gem_obj(struct drm_framebuffer *fb, >> +                    unsigned char plane); >> + >> +void vs_mode_config_init(struct drm_device *dev); >> +#endif /* __VS_FB_H__ */ >