Hi Thomas.
>
> #include "hibmc_drm_drv.h"
> @@ -100,28 +102,41 @@ static int hibmc_plane_atomic_check(struct drm_plane
> *plane,
> static void hibmc_plane_atomic_update(struct drm_plane *plane,
> struct drm_atomic_commit *state)
> {
> + struct hibmc_drm_private *priv = to_hibmc_drm_private(plane->dev);
> struct drm_plane_state *new_state =
> drm_atomic_get_new_plane_state(state, plane);
> + struct drm_shadow_plane_state *shadow_plane_state =
> to_drm_shadow_plane_state(new_state);
> struct drm_framebuffer *fb = new_state->fb;
> + struct drm_plane_state *old_state =
> drm_atomic_get_old_plane_state(state, plane);
> + u32 gpu_addr = 0;
> u32 reg;
> - s64 gpu_addr = 0;
> u32 line_l;
> - struct hibmc_drm_private *priv = to_hibmc_drm_private(plane->dev);
> - struct drm_gem_vram_object *gbo;
>
> - if (!new_state->fb)
> + if (!fb)
> return;
>
> - gbo = drm_gem_vram_of_gem(new_state->fb->obj[0]);
> + if (drm_gem_fb_begin_cpu_access(fb, DMA_FROM_DEVICE) == 0) {
> + struct drm_rect damage;
> + struct drm_atomic_helper_damage_iter iter;
> +
> + drm_atomic_helper_damage_iter_init(&iter, old_state, new_state);
> + drm_atomic_for_each_plane_damage(&iter, &damage) {
> + struct iosys_map dst[DRM_FORMAT_MAX_PLANES] = {
> + IOSYS_MAP_INIT_VADDR_IOMEM(priv->vram +
> gpu_addr),
> + };
>
> - gpu_addr = drm_gem_vram_offset(gbo);
> - if (WARN_ON_ONCE(gpu_addr < 0))
> - return; /* Bug: we didn't pin the BO to VRAM in prepare_fb. */
> + iosys_map_incr(&dst[0],
> + drm_fb_clip_offset(fb->pitches[0],
> fb->format, &damage));
> + drm_fb_memcpy(dst, fb->pitches,
> shadow_plane_state->data, fb, &damage);
> + }
> +
> + drm_gem_fb_end_cpu_access(fb, DMA_FROM_DEVICE);
> + }
This implementation uses shadow planes for damage rect updates. If the entire
plane is updated frequently, could this
lead to a performance drop or frame drops?
Originally, the window system(X11/Wayland) drew directly to the hardware
framebuffer, but now there is an extra copy in
`atomic_update`.
>
> writel(gpu_addr, priv->mmio + HIBMC_CRT_FB_ADDRESS);
>
> reg = drm_format_info_min_pitch(fb->format, 0, fb->width);
>
> - line_l = new_state->fb->pitches[0];
> + line_l = fb->pitches[0];
> writel(HIBMC_FIELD(HIBMC_CRT_FB_WIDTH_WIDTH, reg) |
> HIBMC_FIELD(HIBMC_CRT_FB_WIDTH_OFFS, line_l),
> priv->mmio + HIBMC_CRT_FB_WIDTH);
> @@ -149,13 +164,11 @@ static const struct drm_plane_funcs hibmc_plane_funcs =
> {
> .update_plane = drm_atomic_helper_update_plane,
> .disable_plane = drm_atomic_helper_disable_plane,
> .destroy = drm_plane_cleanup,
> - .reset = drm_atomic_helper_plane_reset,
> - .atomic_duplicate_state = drm_atomic_helper_plane_duplicate_state,
> - .atomic_destroy_state = drm_atomic_helper_plane_destroy_state,
> + DRM_GEM_SHADOW_PLANE_FUNCS,
> };
>
> static const struct drm_plane_helper_funcs hibmc_plane_helper_funcs = {
> - DRM_GEM_VRAM_PLANE_HELPER_FUNCS,
> + DRM_GEM_SHADOW_PLANE_HELPER_FUNCS,
> .atomic_check = hibmc_plane_atomic_check,
> .atomic_update = hibmc_plane_atomic_update,
> };
> @@ -515,6 +528,7 @@ int hibmc_de_init(struct hibmc_drm_private *priv)
> }
>
> drm_plane_helper_add(plane, &hibmc_plane_helper_funcs);
> + drm_plane_enable_fb_damage_clips(plane);
>
> ret = drm_crtc_init_with_planes(dev, crtc, plane,
> NULL, &hibmc_crtc_funcs, NULL);
> diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c
> b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c
> index 99b36de1fe13..19d3193e2f76 100644
> --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c
> +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.c
> @@ -18,9 +18,10 @@
> #include <drm/clients/drm_client_setup.h>
> #include <drm/drm_atomic_helper.h>
> #include <drm/drm_drv.h>
> -#include <drm/drm_fbdev_ttm.h>
> +#include <drm/drm_dumb_buffers.h>
> +#include <drm/drm_fbdev_shmem.h>
> #include <drm/drm_gem_framebuffer_helper.h>
> -#include <drm/drm_gem_vram_helper.h>
> +#include <drm/drm_gem_shmem_helper.h>
> #include <drm/drm_managed.h>
> #include <drm/drm_module.h>
> #include <drm/drm_vblank.h>
> @@ -71,7 +72,13 @@ static irqreturn_t hibmc_dp_interrupt(int irq, void *arg)
> static int hibmc_dumb_create(struct drm_file *file, struct drm_device *dev,
> struct drm_mode_create_dumb *args)
> {
> - return drm_gem_vram_fill_create_dumb(file, dev, 0, 128, args);
> + int ret;
> +
> + ret = drm_mode_size_dumb(dev, args, SZ_128, 0);
> + if (ret)
> + return ret;
> +
> + return drm_gem_shmem_create_with_handle(file, dev, args->size,
> &args->handle);
> }
>
> static const struct drm_driver hibmc_driver = {
> @@ -81,10 +88,9 @@ static const struct drm_driver hibmc_driver = {
> .desc = "hibmc drm driver",
> .major = 1,
> .minor = 0,
> - .debugfs_init = drm_vram_mm_debugfs_init,
> - .dumb_create = hibmc_dumb_create,
> - .dumb_map_offset = drm_gem_ttm_dumb_map_offset,
> - DRM_FBDEV_TTM_DRIVER_OPS,
> + .gem_prime_import = drm_gem_shmem_prime_import_no_map,
> + .dumb_create = hibmc_dumb_create,
> + DRM_FBDEV_SHMEM_DRIVER_OPS,
> };
>
> static int __maybe_unused hibmc_pm_suspend(struct device *dev)
> @@ -106,11 +112,32 @@ static const struct dev_pm_ops hibmc_pm_ops = {
> hibmc_pm_resume)
> };
>
> +static enum drm_mode_status hibmc_mode_config_mode_valid(struct drm_device
> *dev,
> + const struct
> drm_display_mode *mode)
> +{
> + const struct drm_format_info *info =
> + drm_get_format_info(dev, DRM_FORMAT_XRGB8888,
> DRM_FORMAT_MOD_LINEAR);
> + struct hibmc_drm_private *priv = to_hibmc_drm_private(dev);
> + unsigned long max_fb_size = priv->vram_size;
> + u64 pitch;
> +
> + if (drm_WARN_ON_ONCE(dev, !info))
> + return MODE_ERROR; /* driver bug */
> +
> + pitch = drm_format_info_min_pitch(info, 0, mode->hdisplay);
> + if (!pitch)
> + return MODE_BAD_WIDTH;
> + else if (pitch > max_fb_size / mode->vdisplay)
> + return MODE_MEM;
> +
I reviewed the AST and Cirrus implementations, and they include checks for the
maximum size supported by the hardware.
The hardware FB width register has only 14 bits, and the
HIBMC_CRT_FB_WIDTH_WIDTH_MASK macro defines the maximum size as
0x3FFF. Would adding the following check make the code more rigorous?
Also, the pitch requires 128-byte alignment. Should we add a check here, or
does the buffer created by
`hibmc_dumb_create` already ensure that the FB start address is aligned to 128
bytes via `SZ_128`? I’m not entirely sure
about this.
Of course, the original code used `drm_vram_helper_mode_valid` without strictly
checking these hardware limitations.
Thank you for your modification.
Thanks,
Yongbang.
> + return MODE_OK;
> +}
> +
> static const struct drm_mode_config_funcs hibmc_mode_funcs = {
> - .mode_valid = drm_vram_helper_mode_valid,
> + .mode_valid = hibmc_mode_config_mode_valid,
> .atomic_check = drm_atomic_helper_check,
> .atomic_commit = drm_atomic_helper_commit,
> - .fb_create = drm_gem_fb_create,
> + .fb_create = drm_gem_fb_create_with_dirty,
> };
>
> static int hibmc_kms_init(struct hibmc_drm_private *priv)
> @@ -130,7 +157,6 @@ static int hibmc_kms_init(struct hibmc_drm_private *priv)
> dev->mode_config.max_height = 1200;
>
> dev->mode_config.preferred_depth = 24;
> - dev->mode_config.prefer_shadow = 1;
>
> dev->mode_config.funcs = (void *)&hibmc_mode_funcs;
>
> @@ -335,18 +361,22 @@ static int hibmc_load(struct drm_device *dev)
> {
> struct pci_dev *pdev = to_pci_dev(dev->dev);
> struct hibmc_drm_private *priv = to_hibmc_drm_private(dev);
> + resource_size_t vram_base, vram_size;
> int ret;
>
> ret = hibmc_hw_init(priv);
> if (ret)
> return ret;
>
> - ret = drmm_vram_helper_init(dev, pci_resource_start(pdev, 0),
> - pci_resource_len(pdev, 0));
> - if (ret) {
> - drm_err(dev, "Error initializing VRAM MM; %d\n", ret);
> - return ret;
> - }
> + vram_base = pci_resource_start(pdev, 0);
> + vram_size = pci_resource_len(pdev, 0);
> +
> + priv->vram = devm_ioremap_wc(dev->dev, vram_base, vram_size);
> + if (!priv->vram)
> + return -ENOMEM;
> +
> + priv->vram_base = vram_base;
> + priv->vram_size = vram_size;
>
> ret = hibmc_kms_init(priv);
> if (ret)
> diff --git a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h
> b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h
> index cd3a3fca1fe6..dce8572bf63e 100644
> --- a/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h
> +++ b/drivers/gpu/drm/hisilicon/hibmc/hibmc_drm_drv.h
> @@ -38,6 +38,11 @@ struct hibmc_drm_private {
> /* hw */
> void __iomem *mmio;
>
> + /* vram */
> + void __iomem *vram;
> + resource_size_t vram_base;
> + resource_size_t vram_size;
> +
> /* drm */
> struct drm_device dev;
> struct drm_plane primary_plane;
> diff --git a/include/drm/drm_gem_shmem_helper.h
> b/include/drm/drm_gem_shmem_helper.h
> index 5ccdae21b94a..86b967174b60 100644
> --- a/include/drm/drm_gem_shmem_helper.h
> +++ b/include/drm/drm_gem_shmem_helper.h
> @@ -141,6 +141,10 @@ struct sg_table *drm_gem_shmem_get_pages_sgt(struct
> drm_gem_shmem_object *shmem)
> void drm_gem_shmem_print_info(const struct drm_gem_shmem_object *shmem,
> struct drm_printer *p, unsigned int indent);
>
> +int drm_gem_shmem_create_with_handle(struct drm_file *file_priv,
> + struct drm_device *dev, size_t size,
> + uint32_t *handle);
> +
> extern const struct vm_operations_struct drm_gem_shmem_vm_ops;
>
> /*