lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20260126085727.2568958-4-a0987203069@gmail.com>
Date: Mon, 26 Jan 2026 16:57:27 +0800
From: Joey Lu <a0987203069@...il.com>
To: airlied@...il.com,
	simona@...ll.ch,
	maarten.lankhorst@...ux.intel.com,
	mripard@...nel.org,
	tzimmermann@...e.de,
	robh@...nel.org,
	krzk+dt@...nel.org,
	conor+dt@...nel.org
Cc: ychuang3@...oton.com,
	schung@...oton.com,
	yclu4@...oton.com,
	a0987203069@...il.com,
	linux-arm-kernel@...ts.infradead.org,
	dri-devel@...ts.freedesktop.org,
	devicetree@...r.kernel.org,
	linux-kernel@...r.kernel.org
Subject: [PATCH 3/3] drm/nuvoton: add MA35D1 display controller driver

Add DRM driver support for the Display Control Unit (DCU)
found in Nuvoton MA35D1 SoCs.

Signed-off-by: Joey Lu <a0987203069@...il.com>
---
 drivers/gpu/drm/Kconfig                  |   1 +
 drivers/gpu/drm/Makefile                 |   1 +
 drivers/gpu/drm/nuvoton/Kconfig          |  21 +
 drivers/gpu/drm/nuvoton/Makefile         |   7 +
 drivers/gpu/drm/nuvoton/ma35_crtc.c      | 445 +++++++++++
 drivers/gpu/drm/nuvoton/ma35_crtc.h      |  78 ++
 drivers/gpu/drm/nuvoton/ma35_drm.c       | 389 ++++++++++
 drivers/gpu/drm/nuvoton/ma35_drm.h       |  48 ++
 drivers/gpu/drm/nuvoton/ma35_interface.c | 192 +++++
 drivers/gpu/drm/nuvoton/ma35_interface.h |  30 +
 drivers/gpu/drm/nuvoton/ma35_plane.c     | 904 +++++++++++++++++++++++
 drivers/gpu/drm/nuvoton/ma35_plane.h     | 226 ++++++
 drivers/gpu/drm/nuvoton/ma35_regs.h      |  88 +++
 13 files changed, 2430 insertions(+)
 create mode 100644 drivers/gpu/drm/nuvoton/Kconfig
 create mode 100644 drivers/gpu/drm/nuvoton/Makefile
 create mode 100644 drivers/gpu/drm/nuvoton/ma35_crtc.c
 create mode 100644 drivers/gpu/drm/nuvoton/ma35_crtc.h
 create mode 100644 drivers/gpu/drm/nuvoton/ma35_drm.c
 create mode 100644 drivers/gpu/drm/nuvoton/ma35_drm.h
 create mode 100644 drivers/gpu/drm/nuvoton/ma35_interface.c
 create mode 100644 drivers/gpu/drm/nuvoton/ma35_interface.h
 create mode 100644 drivers/gpu/drm/nuvoton/ma35_plane.c
 create mode 100644 drivers/gpu/drm/nuvoton/ma35_plane.h
 create mode 100644 drivers/gpu/drm/nuvoton/ma35_regs.h

diff --git a/drivers/gpu/drm/Kconfig b/drivers/gpu/drm/Kconfig
index a33b90251530..3645255bc458 100644
--- a/drivers/gpu/drm/Kconfig
+++ b/drivers/gpu/drm/Kconfig
@@ -309,6 +309,7 @@ source "drivers/gpu/drm/msm/Kconfig"
 source "drivers/gpu/drm/mxsfb/Kconfig"
 source "drivers/gpu/drm/nouveau/Kconfig"
 source "drivers/gpu/drm/nova/Kconfig"
+source "drivers/gpu/drm/nuvoton/Kconfig"
 source "drivers/gpu/drm/omapdrm/Kconfig"
 source "drivers/gpu/drm/panel/Kconfig"
 source "drivers/gpu/drm/panfrost/Kconfig"
diff --git a/drivers/gpu/drm/Makefile b/drivers/gpu/drm/Makefile
index 0e1c668b46d2..4ded9547d7ff 100644
--- a/drivers/gpu/drm/Makefile
+++ b/drivers/gpu/drm/Makefile
@@ -235,6 +235,7 @@ obj-y			+= solomon/
 obj-$(CONFIG_DRM_SPRD) += sprd/
 obj-$(CONFIG_DRM_LOONGSON) += loongson/
 obj-$(CONFIG_DRM_POWERVR) += imagination/
+obj-$(CONFIG_DRM_MA35) += nuvoton/
 
 # Ensure drm headers are self-contained and pass kernel-doc
 hdrtest-files := \
diff --git a/drivers/gpu/drm/nuvoton/Kconfig b/drivers/gpu/drm/nuvoton/Kconfig
new file mode 100644
index 000000000000..6bb970b9890c
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/Kconfig
@@ -0,0 +1,21 @@
+config DRM_MA35
+	tristate "Nuvoton MA35D1 LCD Display Controller"
+	default ARCH_MA35
+	depends on DRM
+	depends on OF && (ARCH_MA35 || COMPILE_TEST)
+	select DRM_KMS_HELPER
+	select DRM_KMS_DMA_HELPER
+	select DRM_GEM_DMA_HELPER
+	select DRM_BRIDGE
+	select DRM_PANEL_BRIDGE
+	select VIDEOMODE_HELPERS
+	select REGMAP_MMIO
+	help
+	  Choose this option to enable support for the Display Controller Unit (DCU)
+	  found in Nuvoton MA35D1 SoCs.
+
+	  This driver supports the DRM/KMS API for the MA35 display subsystem,
+	  handling display output via hardware composition layers.
+
+	  To compile this driver as a module, choose M here: the module
+	  will be called ma35-drm.
\ No newline at end of file
diff --git a/drivers/gpu/drm/nuvoton/Makefile b/drivers/gpu/drm/nuvoton/Makefile
new file mode 100644
index 000000000000..aac4113106b2
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/Makefile
@@ -0,0 +1,7 @@
+ma35-drm-y += \
+	ma35_drm.o \
+	ma35_plane.o \
+	ma35_crtc.o \
+	ma35_interface.o
+
+obj-$(CONFIG_DRM_MA35)	+= ma35-drm.o
diff --git a/drivers/gpu/drm/nuvoton/ma35_crtc.c b/drivers/gpu/drm/nuvoton/ma35_crtc.c
new file mode 100644
index 000000000000..d168351dcdbe
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/ma35_crtc.c
@@ -0,0 +1,445 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Nuvoton DRM driver
+ *
+ * Copyright (C) 2026 Nuvoton Technology Corp.
+ *
+ * Author: Joey Lu <a0987203069@...il.com>
+ */
+
+#include <linux/clk.h>
+#include <linux/delay.h>
+#include <linux/of.h>
+#include <linux/of_graph.h>
+#include <linux/types.h>
+#include <linux/workqueue.h>
+
+#include <drm/drm_atomic_helper.h>
+#include <drm/drm_crtc.h>
+#include <drm/drm_drv.h>
+#include <drm/drm_gem_dma_helper.h>
+#include <drm/drm_print.h>
+#include <drm/drm_vblank.h>
+
+#include "ma35_drm.h"
+
+#define ma35_crtc(c) \
+	container_of(c, struct ma35_crtc, drm_crtc)
+
+static const struct drm_prop_enum_list ma35_dpi_format[] = {
+	{ MA35_DPI_D16CFG1, "D16CFG1" },
+	{ MA35_DPI_D16CFG2, "D16CFG2" },
+	{ MA35_DPI_D16CFG3, "D16CFG3" },
+	{ MA35_DPI_D18CFG1, "D18CFG1" },
+	{ MA35_DPI_D18CFG2, "D18CFG2" },
+	{ MA35_DPI_D24, "D24" },
+};
+
+static enum drm_mode_status
+ma35_crtc_mode_valid(struct drm_crtc *drm_crtc,
+			const struct drm_display_mode *mode)
+{
+	struct drm_device *drm_dev = drm_crtc->dev;
+	struct drm_mode_config *mode_config = &drm_dev->mode_config;
+
+	/* check drm_mode_status for some limitations */
+	if (mode->flags & DRM_MODE_FLAG_INTERLACE)
+		return MODE_NO_INTERLACE;
+
+	if (mode->hdisplay > mode_config->max_width || mode->hdisplay < mode_config->min_width)
+		return MODE_BAD_HVALUE;
+
+	if (mode->vdisplay > mode_config->max_height || mode->vdisplay < mode_config->min_height)
+		return MODE_BAD_VVALUE;
+
+	if (mode->clock > MA35_MAX_PIXEL_CLK)
+		return MODE_CLOCK_HIGH;
+
+	return MODE_OK;
+}
+
+static int ma35_crtc_atomic_check(struct drm_crtc *drm_crtc,
+					 struct drm_atomic_state *state)
+{
+	struct ma35_drm *priv = ma35_drm(drm_crtc->dev);
+	struct drm_crtc_state *crtc_state = drm_atomic_get_new_crtc_state(state, drm_crtc);
+	struct drm_display_mode *mode = &crtc_state->mode;
+	struct drm_display_mode *adjusted_mode = &crtc_state->adjusted_mode;
+	int clk_rate;
+
+	if (mode->clock > MA35_MAX_PIXEL_CLK)
+		return MODE_CLOCK_HIGH;
+
+	/* check rounded pixel clock */
+	clk_rate = clk_round_rate(priv->dcupclk, mode->clock * 1000);
+	if (clk_rate <= 0)
+		return MODE_CLOCK_RANGE;
+
+	adjusted_mode->clock = DIV_ROUND_UP(clk_rate, 1000);
+
+	return 0;
+}
+
+static void ma35_crtc_atomic_enable(struct drm_crtc *drm_crtc,
+				       struct drm_atomic_state *state)
+{
+	struct ma35_crtc *crtc = ma35_crtc(drm_crtc);
+	struct ma35_drm *priv = ma35_drm(drm_crtc->dev);
+	struct drm_crtc_state *new_state =
+		drm_atomic_get_new_crtc_state(state, drm_crtc);
+	struct drm_display_mode *mode = &new_state->adjusted_mode;
+	struct ma35_interface *interface = priv->interface;
+	struct drm_color_lut *lut;
+	int i, size;
+	u32 reg;
+
+	/* Timings */
+	reg = FIELD_PREP(MA35_DISPLAY_TOTAL_MASK, mode->htotal) |
+		  FIELD_PREP(MA35_DISPLAY_ACTIVE_MASK, mode->hdisplay);
+	regmap_write(priv->regmap, MA35_HDISPLAY, reg);
+
+	reg = MA35_SYNC_PULSE_ENABLE |
+		  FIELD_PREP(MA35_SYNC_START_MASK, mode->hsync_start) |
+		  FIELD_PREP(MA35_SYNC_END_MASK, mode->hsync_end);
+	if (mode->flags & DRM_MODE_FLAG_NHSYNC)
+		reg |= MA35_SYNC_POLARITY_BIT;
+	regmap_write(priv->regmap, MA35_HSYNC, reg);
+
+	reg = FIELD_PREP(MA35_DISPLAY_TOTAL_MASK, mode->vtotal) |
+		  FIELD_PREP(MA35_DISPLAY_ACTIVE_MASK, mode->vdisplay);
+	regmap_write(priv->regmap, MA35_VDISPLAY, reg);
+
+	reg = MA35_SYNC_PULSE_ENABLE |
+		  FIELD_PREP(MA35_SYNC_START_MASK, mode->vsync_start) |
+		  FIELD_PREP(MA35_SYNC_END_MASK, mode->vsync_end);
+	if (mode->flags & DRM_MODE_FLAG_NVSYNC)
+		reg |= MA35_SYNC_POLARITY_BIT;
+	regmap_write(priv->regmap, MA35_VSYNC, reg);
+
+	/* Signals */
+	reg = MA35_PANEL_DATA_ENABLE_ENABLE | MA35_PANEL_DATA_ENABLE |
+		  MA35_PANEL_DATA_CLOCK_ENABLE;
+	if (interface->bus_flags & DRM_BUS_FLAG_DE_LOW)
+		reg |= MA35_PANEL_DATA_ENABLE_POLARITY;
+
+	if (interface->bus_flags & DRM_BUS_FLAG_PIXDATA_DRIVE_NEGEDGE)
+		reg |= MA35_PANEL_DATA_POLARITY;
+	regmap_write(priv->regmap, MA35_PANEL_CONFIG, reg);
+
+	/* Gamma */
+	if (new_state->gamma_lut) {
+		if (new_state->color_mgmt_changed) {
+			lut = new_state->gamma_lut->data;
+			size = new_state->gamma_lut->length / sizeof(struct drm_color_lut);
+
+			for (i = 0; i < size; i++) {
+				regmap_write(priv->regmap, MA35_GAMMA_INDEX, i);
+				/* shift DRM gamma 16-bit values to 10-bit */
+				reg = FIELD_PREP(MA35_GAMMA_RED_MASK, lut[i].red >> 6) |
+					  FIELD_PREP(MA35_GAMMA_GREEN_MASK, lut[i].green >> 6) |
+					  FIELD_PREP(MA35_GAMMA_BLUE_MASK, lut[i].blue >> 6);
+				regmap_write(priv->regmap, MA35_GAMMA_DATA, reg);
+			}
+		}
+		/* Enable gamma */
+		regmap_update_bits(priv->regmap, MA35_FRAMEBUFFER_CONFIG,
+						   MA35_PRIMARY_GAMMA, MA35_PRIMARY_GAMMA);
+	} else {
+		/* Disable gamma */
+		regmap_update_bits(priv->regmap, MA35_FRAMEBUFFER_CONFIG,
+						   MA35_PRIMARY_GAMMA, 0);
+	}
+
+	/* DPI format */
+	reg = FIELD_PREP(MA35_DPI_FORMAT_MASK, crtc->dpi_format);
+	regmap_write(priv->regmap, MA35_DPI_CONFIG, reg);
+
+	/* Dither */
+	if (crtc->dither_enable) {
+		for (i = 0, reg = 0; i < MA35_DITHER_TABLE_ENTRY / 2; i++)
+			reg |= (crtc->dither_depth & MA35_DITHER_TABLE_MASK) << (i * 4);
+
+		regmap_write(priv->regmap, MA35_DISPLAY_DITHER_TABLE_LOW, reg);
+		regmap_write(priv->regmap, MA35_DISPLAY_DITHER_TABLE_HIGH, reg);
+		regmap_write(priv->regmap, MA35_DISPLAY_DITHER_CONFIG, MA35_DITHER_ENABLE);
+	} else {
+		regmap_write(priv->regmap, MA35_DISPLAY_DITHER_CONFIG, 0);
+	}
+
+	drm_crtc_vblank_on(drm_crtc);
+}
+
+static void ma35_crtc_atomic_disable(struct drm_crtc *drm_crtc,
+					struct drm_atomic_state *state)
+{
+	struct ma35_drm *priv = ma35_drm(drm_crtc->dev);
+	struct drm_device *drm_dev = drm_crtc->dev;
+
+	drm_crtc_vblank_off(drm_crtc);
+
+	/* Disable and clear CRTC bits. */
+	regmap_update_bits(priv->regmap, MA35_PANEL_CONFIG,
+					   MA35_PANEL_DATA_ENABLE_ENABLE, 0);
+	regmap_update_bits(priv->regmap, MA35_FRAMEBUFFER_CONFIG,
+					   MA35_PRIMARY_GAMMA, 0);
+	regmap_write(priv->regmap, MA35_DISPLAY_DITHER_CONFIG, 0);
+
+	/* Consume any leftover event since vblank is now disabled. */
+	if (drm_crtc->state->event && !drm_crtc->state->active) {
+		spin_lock_irq(&drm_dev->event_lock);
+
+		drm_crtc_send_vblank_event(drm_crtc, drm_crtc->state->event);
+		drm_crtc->state->event = NULL;
+		spin_unlock_irq(&drm_dev->event_lock);
+	}
+}
+
+static void ma35_crtc_atomic_flush(struct drm_crtc *drm_crtc,
+	struct drm_atomic_state *state)
+{
+	spin_lock_irq(&drm_crtc->dev->event_lock);
+	if (drm_crtc->state->event) {
+		if (drm_crtc_vblank_get(drm_crtc) == 0)
+			drm_crtc_arm_vblank_event(drm_crtc, drm_crtc->state->event);
+		else
+			drm_crtc_send_vblank_event(drm_crtc, drm_crtc->state->event);
+
+		drm_crtc->state->event = NULL;
+	}
+	spin_unlock_irq(&drm_crtc->dev->event_lock);
+}
+
+static bool ma35_crtc_get_scanout_position(struct drm_crtc *drm_crtc,
+					   bool in_vblank_irq,
+					   int *vpos,
+					   int *hpos,
+					   ktime_t *stime,
+					   ktime_t *etime,
+					   const struct drm_display_mode *mode)
+{
+	struct ma35_drm *priv = ma35_drm(drm_crtc->dev);
+	u32 reg;
+
+	if (stime)
+		*stime = ktime_get();
+
+	regmap_read(priv->regmap, MA35_DISPLAY_CURRENT_LOCATION, &reg);
+
+	*hpos = FIELD_GET(MA35_DISPLAY_CURRENT_X, reg);
+	*vpos = FIELD_GET(MA35_DISPLAY_CURRENT_Y, reg);
+
+	if (etime)
+		*etime = ktime_get();
+
+	return true;
+}
+
+static const struct drm_crtc_helper_funcs ma35_crtc_helper_funcs = {
+	.mode_valid		= ma35_crtc_mode_valid,
+	.atomic_check		= ma35_crtc_atomic_check,
+	.atomic_enable		= ma35_crtc_atomic_enable,
+	.atomic_disable		= ma35_crtc_atomic_disable,
+	.atomic_flush		= ma35_crtc_atomic_flush,
+	.get_scanout_position	= ma35_crtc_get_scanout_position,
+};
+
+static int ma35_crtc_enable_vblank(struct drm_crtc *drm_crtc)
+{
+	struct ma35_drm *priv = ma35_drm(drm_crtc->dev);
+
+	regmap_write(priv->regmap, MA35_DISPLAY_INTRENABLE,
+			  MA35_CRTC_VBLANK);
+
+	return 0;
+}
+
+static void ma35_crtc_disable_vblank(struct drm_crtc *drm_crtc)
+{
+	struct ma35_drm *priv = ma35_drm(drm_crtc->dev);
+
+	regmap_write(priv->regmap, MA35_DISPLAY_INTRENABLE, 0);
+}
+
+static u32 ma35_crtc_get_vblank_counter(struct drm_crtc *drm_crtc)
+{
+	struct ma35_drm *priv = ma35_drm(drm_crtc->dev);
+
+	return atomic_read(&priv->crtc->vblank_counter);
+}
+
+static int ma35_crtc_gamma_set(struct drm_crtc *drm_crtc,
+		  u16 *r, u16 *g, u16 *b, uint32_t size,
+		  struct drm_modeset_acquire_ctx *ctx)
+{
+	struct ma35_drm *priv = ma35_drm(drm_crtc->dev);
+	u32 reg;
+	int i;
+
+	if (size != MA35_GAMMA_TABLE_SIZE)
+		return -EINVAL;
+
+	regmap_write(priv->regmap, MA35_GAMMA_INDEX, 0); // auto increment
+
+	for (i = 0; i < size; i++) {
+		reg = FIELD_PREP(MA35_GAMMA_RED_MASK, r[i]) |
+			  FIELD_PREP(MA35_GAMMA_GREEN_MASK, g[i]) |
+			  FIELD_PREP(MA35_GAMMA_BLUE_MASK, b[i]);
+		regmap_write(priv->regmap, MA35_GAMMA_DATA, reg);
+	}
+
+	return 0;
+}
+
+static int ma35_crtc_atomic_set_property(struct drm_crtc *drm_crtc,
+					 struct drm_crtc_state *state,
+					 struct drm_property *property,
+					 uint64_t value)
+{
+	struct ma35_crtc *crtc = ma35_crtc(drm_crtc);
+
+	if (property == crtc->dpi_format_prop)
+		crtc->dpi_format = value;
+	else if (property == crtc->dither_enable_prop)
+		crtc->dither_enable = value;
+	else if (property == crtc->dither_depth_prop)
+		crtc->dither_depth = value;
+	else
+		return -EINVAL;
+
+	return 0;
+}
+
+static int ma35_crtc_atomic_get_property(struct drm_crtc *drm_crtc,
+					 const struct drm_crtc_state *state,
+					 struct drm_property *property,
+					 uint64_t *value)
+{
+	struct ma35_crtc *crtc = ma35_crtc(drm_crtc);
+
+	if (property == crtc->dpi_format_prop)
+		*value = crtc->dpi_format;
+	else if (property == crtc->dither_enable_prop)
+		*value = crtc->dither_enable;
+	else if (property == crtc->dither_depth_prop)
+		*value = crtc->dither_depth;
+	else
+		return -EINVAL;
+
+	return 0;
+}
+
+static const struct drm_crtc_funcs ma35_crtc_funcs = {
+	.reset			= drm_atomic_helper_crtc_reset,
+	.destroy		= drm_crtc_cleanup,
+	.set_config		= drm_atomic_helper_set_config,
+	.page_flip		= drm_atomic_helper_page_flip,
+	.atomic_duplicate_state	= drm_atomic_helper_crtc_duplicate_state,
+	.atomic_destroy_state	= drm_atomic_helper_crtc_destroy_state,
+	.enable_vblank		= ma35_crtc_enable_vblank,
+	.disable_vblank		= ma35_crtc_disable_vblank,
+	.get_vblank_counter = ma35_crtc_get_vblank_counter,
+	.gamma_set      = ma35_crtc_gamma_set,
+	.atomic_set_property = ma35_crtc_atomic_set_property,
+	.atomic_get_property = ma35_crtc_atomic_get_property,
+};
+
+void ma35_crtc_vblank_handler(struct ma35_drm *priv)
+{
+	struct ma35_crtc *crtc = priv->crtc;
+
+	if (!crtc)
+		return;
+
+	atomic_inc(&crtc->vblank_counter);
+
+	drm_crtc_handle_vblank(&crtc->drm_crtc);
+}
+
+static int ma35_crtc_create_properties(struct ma35_drm *priv)
+{
+	struct drm_device *drm_dev = &priv->drm_dev;
+	struct ma35_crtc *crtc = priv->crtc;
+	struct drm_crtc *drm_crtc = &crtc->drm_crtc;
+
+	crtc->dpi_format_prop = drm_property_create_enum(drm_dev, 0,
+						"dpi-format",
+						ma35_dpi_format,
+						ARRAY_SIZE(ma35_dpi_format));
+	if (!crtc->dpi_format_prop) {
+		drm_err(drm_dev, "Failed to create dpi format property\n");
+		return -ENOMEM;
+	}
+	drm_object_attach_property(&drm_crtc->base, crtc->dpi_format_prop, MA35_DPI_D24);
+	crtc->dpi_format = MA35_DPI_D24;
+
+	crtc->dither_enable_prop = drm_property_create_bool(drm_dev, 0, "dither-enable");
+	if (!crtc->dither_enable_prop) {
+		drm_err(drm_dev, "Failed to create dither enable property\n");
+		return -ENOMEM;
+	}
+	drm_object_attach_property(&drm_crtc->base, crtc->dither_enable_prop, false);
+	crtc->dither_enable = false;
+
+	crtc->dither_depth_prop = drm_property_create_range(drm_dev, 0, "dither-depth",
+							0, 0xf);
+	if (!crtc->dither_depth_prop) {
+		drm_err(drm_dev, "Failed to create dither depth property\n");
+		return -ENOMEM;
+	}
+	drm_object_attach_property(&drm_crtc->base, crtc->dither_depth_prop, 0);
+	crtc->dither_depth = 0;
+
+	return 0;
+}
+
+int ma35_crtc_init(struct ma35_drm *priv)
+{
+	struct drm_device *drm_dev = &priv->drm_dev;
+	struct device *dev = drm_dev->dev;
+	struct ma35_crtc *crtc;
+	struct ma35_layer *layer_primary, *layer_cursor;
+	struct drm_plane *cursor_plane = NULL;
+	int ret;
+
+	crtc = devm_kzalloc(dev, sizeof(*crtc), GFP_KERNEL);
+	if (!crtc)
+		return -ENOMEM;
+
+	priv->crtc = crtc;
+	atomic_set(&crtc->vblank_counter, 0);
+
+	layer_primary = ma35_layer_get_from_type(priv, DRM_PLANE_TYPE_PRIMARY);
+	if (!layer_primary) {
+		drm_err(drm_dev, "Failed to get primary layer\n");
+		return -EINVAL;
+	}
+
+	layer_cursor = ma35_layer_get_from_type(priv, DRM_PLANE_TYPE_CURSOR);
+	if (layer_cursor)
+		cursor_plane = &layer_cursor->drm_plane;
+
+	/* attach primary and cursor */
+	ret = drm_crtc_init_with_planes(drm_dev, &crtc->drm_crtc,
+					&layer_primary->drm_plane, cursor_plane,
+					&ma35_crtc_funcs, NULL);
+	if (ret) {
+		drm_err(drm_dev, "Failed to initialize CRTC\n");
+		return ret;
+	}
+
+	/* attach overlay */
+	ma35_overlay_attach_crtc(priv);
+
+	/* dither & gamma */
+	ret = ma35_crtc_create_properties(priv);
+	if (ret)
+		return ret;
+	ret = drm_mode_crtc_set_gamma_size(&crtc->drm_crtc, MA35_GAMMA_TABLE_SIZE);
+	if (ret)
+		return ret;
+	drm_crtc_enable_color_mgmt(&crtc->drm_crtc, 0, false, MA35_GAMMA_TABLE_SIZE);
+
+	drm_crtc_helper_add(&crtc->drm_crtc, &ma35_crtc_helper_funcs);
+
+	return 0;
+}
diff --git a/drivers/gpu/drm/nuvoton/ma35_crtc.h b/drivers/gpu/drm/nuvoton/ma35_crtc.h
new file mode 100644
index 000000000000..71a41da21df7
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/ma35_crtc.h
@@ -0,0 +1,78 @@
+/* SPDX-License-Identifier: GPL-2.0+ */
+/*
+ * Nuvoton DRM driver
+ *
+ * Copyright (C) 2026 Nuvoton Technology Corp.
+ *
+ * Author: Joey Lu <a0987203069@...il.com>
+ */
+
+#ifndef _MA35_CRTC_H_
+#define _MA35_CRTC_H_
+
+#include <drm/drm_crtc.h>
+
+struct drm_pending_vblank_event;
+struct ma35_drm;
+
+enum ma35_dpi_format_enum {
+	MA35_DPI_D16CFG1,
+	MA35_DPI_D16CFG2,
+	MA35_DPI_D16CFG3,
+	MA35_DPI_D18CFG1,
+	MA35_DPI_D18CFG2,
+	MA35_DPI_D24,
+};
+
+#define MA35_DPI_FORMAT_MASK GENMASK(2, 0)
+
+struct ma35_crtc {
+	struct drm_crtc drm_crtc;
+	struct drm_property *dpi_format_prop;
+	struct drm_property *dither_depth_prop;
+	struct drm_property *dither_enable_prop;
+	atomic_t vblank_counter;
+	u32 dpi_format;
+	u16 dither_depth;
+	bool dither_enable;
+};
+
+#define MA35_DEFAULT_CRTC_ID	0
+
+#define MA35_MAX_PIXEL_CLK		150000
+
+#define MA35_GAMMA_TABLE_SIZE	256
+#define MA35_GAMMA_RED_MASK		GENMASK(29, 20)
+#define MA35_GAMMA_GREEN_MASK	GENMASK(19, 10)
+#define MA35_GAMMA_BLUE_MASK	GENMASK(9, 0)
+
+#define MA35_DITHER_TABLE_ENTRY	16
+#define MA35_DITHER_ENABLE		BIT(31)
+#define MA35_DITHER_TABLE_MASK	GENMASK(3, 0)
+
+#define MA35_CRTC_VBLANK		BIT(0)
+
+#define MA35_DEBUG_COUNTER_MASK		GENMASK(31, 0)
+
+#define MA35_PANEL_DATA_ENABLE_ENABLE	BIT(0)
+#define MA35_PANEL_DATA_ENABLE_POLARITY	BIT(1)
+#define MA35_PANEL_DATA_ENABLE			BIT(4)
+#define MA35_PANEL_DATA_POLARITY		BIT(5)
+#define MA35_PANEL_DATA_CLOCK_ENABLE	BIT(8)
+#define MA35_PANEL_DATA_CLOCK_POLARITY	BIT(9)
+
+#define MA35_DISPLAY_TOTAL_MASK		GENMASK(30, 16)
+#define MA35_DISPLAY_ACTIVE_MASK	GENMASK(14, 0)
+
+#define MA35_SYNC_POLARITY_BIT	BIT(31)
+#define MA35_SYNC_PULSE_ENABLE	BIT(30)
+#define MA35_SYNC_END_MASK		GENMASK(29, 15)
+#define MA35_SYNC_START_MASK	GENMASK(14, 0)
+
+#define MA35_DISPLAY_CURRENT_X	GENMASK(15, 0)
+#define MA35_DISPLAY_CURRENT_Y	GENMASK(31, 16)
+
+void ma35_crtc_vblank_handler(struct ma35_drm *priv);
+int ma35_crtc_init(struct ma35_drm *priv);
+
+#endif
diff --git a/drivers/gpu/drm/nuvoton/ma35_drm.c b/drivers/gpu/drm/nuvoton/ma35_drm.c
new file mode 100644
index 000000000000..b675f6d346b1
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/ma35_drm.c
@@ -0,0 +1,389 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Nuvoton DRM driver
+ *
+ * Copyright (C) 2026 Nuvoton Technology Corp.
+ *
+ * Author: Joey Lu <a0987203069@...il.com>
+ */
+
+#include <linux/bitfield.h>
+#include <linux/clk.h>
+#include <linux/mfd/syscon.h>
+#include <linux/module.h>
+#include <linux/of.h>
+#include <linux/of_address.h>
+#include <linux/of_reserved_mem.h>
+#include <linux/platform_device.h>
+#include <linux/regmap.h>
+#include <linux/types.h>
+
+#include <drm/clients/drm_client_setup.h>
+#include <drm/drm_atomic_helper.h>
+#include <drm/drm_drv.h>
+#include <drm/drm_fb_helper.h>
+#include <drm/drm_fbdev_dma.h>
+#include <drm/drm_gem_dma_helper.h>
+#include <drm/drm_gem_framebuffer_helper.h>
+#include <drm/drm_print.h>
+#include <drm/drm_probe_helper.h>
+#include <drm/drm_vblank.h>
+
+#include "ma35_drm.h"
+
+DEFINE_DRM_GEM_DMA_FOPS(ma35_drm_fops);
+
+static int ma35_drm_gem_dma_dumb_create(struct drm_file *file_priv,
+					   struct drm_device *drm_dev,
+					   struct drm_mode_create_dumb *args)
+{
+	struct drm_mode_config *mode_config = &drm_dev->mode_config;
+	u32 pixel_align;
+
+	if (args->width < mode_config->min_width ||
+		args->height < mode_config->min_height)
+		return -EINVAL;
+
+	/* check for alignment */
+	pixel_align = MA35_DISPLAY_ALIGN_PIXELS * args->bpp / 8;
+	args->pitch = ALIGN(args->width * args->bpp / 8, pixel_align);
+
+	return drm_gem_dma_dumb_create_internal(file_priv, drm_dev, args);
+}
+
+static struct drm_driver ma35_drm_driver = {
+	.driver_features	= DRIVER_GEM | DRIVER_MODESET | DRIVER_ATOMIC |
+						  DRIVER_CURSOR_HOTSPOT,
+
+	.fops				= &ma35_drm_fops,
+	.name				= "ma35-drm",
+	.desc				= "Nuvoton MA35 series DRM driver",
+	.major				= DRIVER_MAJOR,
+	.minor				= DRIVER_MINOR,
+
+	DRM_GEM_DMA_DRIVER_OPS_VMAP_WITH_DUMB_CREATE(ma35_drm_gem_dma_dumb_create),
+};
+
+static const struct regmap_config ma35_drm_regmap_config = {
+	.reg_bits	= 32,
+	.val_bits	= 32,
+	.reg_stride	= 4,
+	.max_register	= 0x2000,
+	.name		= "ma35-drm",
+};
+
+static irqreturn_t ma35_drm_irq_handler(int irq, void *data)
+{
+	struct ma35_drm *priv = data;
+	irqreturn_t ret = IRQ_NONE;
+	u32 stat = 0;
+
+	/* Get pending interrupt sources (RO) */
+	regmap_read(priv->regmap, MA35_INT_STATE, &stat);
+
+	if (stat & MA35_INT_STATE_DISP0) {
+		ma35_crtc_vblank_handler(priv);
+		ret = IRQ_HANDLED;
+	}
+
+	return ret;
+}
+
+static const struct drm_mode_config_funcs ma35_mode_config_funcs = {
+	.fb_create		= drm_gem_fb_create,
+	.atomic_check		= drm_atomic_helper_check,
+	.atomic_commit		= drm_atomic_helper_commit,
+};
+
+static const struct drm_mode_config_helper_funcs ma35_mode_config_helper_funcs = {
+	.atomic_commit_tail = drm_atomic_helper_commit_tail,
+};
+
+static int ma35_mode_init(struct ma35_drm *priv)
+{
+	struct drm_device *drm_dev = &priv->drm_dev;
+	struct drm_mode_config *mode_config = &drm_dev->mode_config;
+	int ret;
+
+	ret = drmm_mode_config_init(drm_dev);
+	if (ret) {
+		drm_err(drm_dev, "Failed to init mode config\n");
+		return -EINVAL;
+	}
+
+	drm_dev->max_vblank_count = MA35_DEBUG_COUNTER_MASK;
+	ret = drm_vblank_init(drm_dev, 1);
+	if (ret) {
+		drm_err(drm_dev, "Failed to initialize vblank\n");
+		return ret;
+	}
+
+	mode_config->min_width = 32;
+	mode_config->max_width = 1920;
+	mode_config->min_height = 1;
+	mode_config->max_height = 1080;
+	mode_config->preferred_depth = 24;
+	mode_config->cursor_width = MA35_CURSOR_WIDTH;
+	mode_config->cursor_height = MA35_CURSOR_HEIGHT;
+	mode_config->funcs = &ma35_mode_config_funcs;
+	mode_config->helper_private = &ma35_mode_config_helper_funcs;
+
+	return 0;
+}
+
+static void ma35_mode_fini(struct ma35_drm *priv)
+{
+	struct drm_device *drm_dev = &priv->drm_dev;
+
+	drm_kms_helper_poll_fini(drm_dev);
+}
+
+static int ma35_clocks_prepare(struct ma35_drm *priv)
+{
+	struct drm_device *drm_dev = &priv->drm_dev;
+	struct device *dev = drm_dev->dev;
+	int ret;
+
+	priv->dcuclk = devm_clk_get(dev, "dcu_gate");
+	if (IS_ERR(priv->dcuclk)) {
+		dev_err(dev, "Failed to get display core clock\n");
+		return PTR_ERR(priv->dcuclk);
+	}
+
+	ret = clk_prepare_enable(priv->dcuclk);
+	if (ret) {
+		dev_err(dev, "Failed to enable display core clock\n");
+		return ret;
+	}
+
+	priv->dcupclk = devm_clk_get(dev, "dcup_div");
+	if (IS_ERR(priv->dcupclk)) {
+		dev_err(dev, "Failed to get display pixel clock\n");
+		return PTR_ERR(priv->dcupclk);
+	}
+
+	ret = clk_prepare_enable(priv->dcupclk);
+	if (ret) {
+		dev_err(dev, "Failed to enable display pixel clock\n");
+		return ret;
+	}
+
+	return 0;
+}
+
+static int ma35_clocks_unprepare(struct ma35_drm *priv)
+{
+	struct clk **clocks[] = {
+		&priv->dcuclk,
+		&priv->dcupclk,
+	};
+	unsigned int i;
+
+	for (i = 0; i < ARRAY_SIZE(clocks); i++) {
+		if (!*clocks[i])
+			continue;
+
+		clk_disable_unprepare(*clocks[i]);
+		*clocks[i] = NULL;
+	}
+
+	return 0;
+}
+
+static int ma35_drm_probe(struct platform_device *pdev)
+{
+	struct device_node *of_node = pdev->dev.of_node;
+	struct device *dev = &pdev->dev;
+	struct device_node *mem_node;
+	struct resource res;
+	struct ma35_drm *priv;
+	struct drm_device *drm_dev;
+	void __iomem *base;
+	struct regmap *regmap = NULL;
+	int irq;
+	int ret;
+
+	/* Check for reserved memory. Fallback to dynamic allocation if undefined */
+	mem_node = of_parse_phandle(of_node, "memory-region", 0);
+	if (mem_node) {
+		ret = of_address_to_resource(mem_node, 0, &res);
+		if (ret) {
+			dev_err(dev, "Failed to parse reserved memory resource: %d\n", ret);
+			of_node_put(mem_node);
+			return ret;
+		}
+		of_node_put(mem_node);
+		dev_info(dev, "registering reserved memory %pR\n", &res);
+
+		ret = of_reserved_mem_device_init(dev);
+		if (ret && ret != -ENODEV) {
+			dev_err(dev, "Failed to init memory region\n");
+			goto error_early;
+		}
+	}
+
+	base = devm_platform_ioremap_resource(pdev, 0);
+	if (IS_ERR(base)) {
+		dev_err(dev, "Failed to map I/O base\n");
+		ret = PTR_ERR(base);
+		goto error_reserved_mem;
+	}
+	regmap = devm_regmap_init_mmio(dev, base, &ma35_drm_regmap_config);
+	if (IS_ERR(regmap)) {
+		dev_err(dev, "Failed to create regmap for I/O\n");
+		ret = PTR_ERR(regmap);
+		goto error_reserved_mem;
+	}
+
+	irq = platform_get_irq(pdev, 0);
+	if (irq < 0) {
+		ret = -ENODEV;
+		goto error_reserved_mem;
+	}
+
+	priv = devm_drm_dev_alloc(dev, &ma35_drm_driver,
+				     struct ma35_drm, drm_dev);
+	if (IS_ERR(priv)) {
+		ret = PTR_ERR(priv);
+		goto error_reserved_mem;
+	}
+
+	platform_set_drvdata(pdev, priv);
+	drm_dev = &priv->drm_dev;
+	priv->regmap = regmap;
+	INIT_LIST_HEAD(&priv->layers_list);
+
+	ret = ma35_clocks_prepare(priv);
+	if (ret) {
+		drm_err(drm_dev, "Failed to prepare clocks\n");
+		goto error_reserved_mem;
+	}
+
+	ret = devm_request_irq(dev, irq, ma35_drm_irq_handler, 0,
+			       dev_name(dev), priv);
+	if (ret) {
+		drm_err(drm_dev, "Failed to request IRQ\n");
+		goto error_clocks;
+	}
+
+	/* modeset */
+	ret = ma35_mode_init(priv);
+	if (ret) {
+		drm_err(drm_dev, "Failed to initialize KMS\n");
+		goto error_clocks;
+	}
+
+	/* plane */
+	ret = ma35_plane_init(priv);
+	if (ret) {
+		drm_err(drm_dev, "Failed to initialize layers\n");
+		goto error_clocks;
+	}
+
+	/* crtc */
+	ret = ma35_crtc_init(priv);
+	if (ret) {
+		drm_err(drm_dev, "Failed to initialize CRTC\n");
+		goto error_clocks;
+	}
+
+	/* interface */
+	ret = ma35_interface_init(priv);
+	if (ret) {
+		if (ret != -EPROBE_DEFER)
+			drm_err(drm_dev, "Failed to initialize interface\n");
+
+		goto error_clocks;
+	}
+
+	drm_mode_config_reset(drm_dev);
+
+	ret = drm_dev_register(drm_dev, 0);
+	if (ret) {
+		drm_err(drm_dev, "Failed to register DRM device\n");
+		goto error_mode;
+	}
+
+	drm_client_setup(drm_dev, NULL);
+
+	return 0;
+
+error_mode:
+	ma35_mode_fini(priv);
+
+error_clocks:
+	ma35_clocks_unprepare(priv);
+
+error_reserved_mem:
+	of_reserved_mem_device_release(dev);
+
+error_early:
+	return ret;
+}
+
+static void ma35_drm_remove(struct platform_device *pdev)
+{
+	struct ma35_drm *priv = platform_get_drvdata(pdev);
+	struct device *dev = &pdev->dev;
+	struct drm_device *drm_dev = &priv->drm_dev;
+
+	drm_dev_unregister(drm_dev);
+	drm_atomic_helper_shutdown(drm_dev);
+
+	ma35_mode_fini(priv);
+
+	ma35_clocks_unprepare(priv);
+
+	of_reserved_mem_device_release(dev);
+}
+
+static void ma35_drm_shutdown(struct platform_device *pdev)
+{
+	struct ma35_drm *priv = platform_get_drvdata(pdev);
+	struct drm_device *drm_dev = &priv->drm_dev;
+
+	drm_atomic_helper_shutdown(drm_dev);
+}
+
+static __maybe_unused int ma35_drm_suspend(struct device *dev)
+{
+	struct ma35_drm *priv = dev_get_drvdata(dev);
+	struct drm_device *drm_dev = &priv->drm_dev;
+
+	return drm_mode_config_helper_suspend(drm_dev);
+}
+
+static __maybe_unused int ma35_drm_resume(struct device *dev)
+{
+	struct ma35_drm *priv = dev_get_drvdata(dev);
+	struct drm_device *drm_dev = &priv->drm_dev;
+
+	return drm_mode_config_helper_resume(drm_dev);
+}
+
+static const struct of_device_id ma35_drm_of_table[] = {
+	{ .compatible = "nuvoton,ma35d1-dcu" },
+	{},
+};
+MODULE_DEVICE_TABLE(of, ma35_drm_of_table);
+
+static const struct dev_pm_ops ma35_pm_ops = {
+	SET_SYSTEM_SLEEP_PM_OPS(ma35_drm_suspend, ma35_drm_resume)
+};
+
+static struct platform_driver ma35_drm_platform_driver = {
+	.probe		= ma35_drm_probe,
+	.remove		= ma35_drm_remove,
+	.shutdown	= ma35_drm_shutdown,
+	.driver		= {
+		.name		= "ma35-drm",
+		.of_match_table	= ma35_drm_of_table,
+		.pm = &ma35_pm_ops,
+	},
+};
+
+module_platform_driver(ma35_drm_platform_driver);
+
+MODULE_AUTHOR("Joey Lu <a0987203069@...il.com>");
+MODULE_DESCRIPTION("Nuvoton MA35 series DRM driver");
+MODULE_LICENSE("GPL");
diff --git a/drivers/gpu/drm/nuvoton/ma35_drm.h b/drivers/gpu/drm/nuvoton/ma35_drm.h
new file mode 100644
index 000000000000..68da6b11a323
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/ma35_drm.h
@@ -0,0 +1,48 @@
+/* SPDX-License-Identifier: GPL-2.0+ */
+/*
+ * Nuvoton DRM driver
+ *
+ * Copyright (C) 2026 Nuvoton Technology Corp.
+ *
+ * Author: Joey Lu <a0987203069@...il.com>
+ */
+
+#ifndef _MA35_DRM_H_
+#define _MA35_DRM_H_
+
+#include <linux/regmap.h>
+#include <linux/types.h>
+#include <drm/drm_device.h>
+
+#include "ma35_regs.h"
+#include "ma35_plane.h"
+#include "ma35_crtc.h"
+#include "ma35_interface.h"
+
+#define DRIVER_MAJOR	1
+#define DRIVER_MINOR	0
+
+#define MA35_INT_STATE_DISP0	BIT(0)
+
+#define MA35_DISPLAY_ALIGN_PIXELS	32
+#define MA35_DISPLAY_PREFER_DEPTH	32
+
+#define MA35_CURSOR_WIDTH	32
+#define MA35_CURSOR_HEIGHT	32
+
+#define MA35_DISPLAY_MAX_ZPOS	3
+
+#define ma35_drm(d) \
+	container_of(d, struct ma35_drm, drm_dev)
+
+struct ma35_drm {
+	struct drm_device drm_dev;
+	struct regmap *regmap;
+	struct list_head layers_list;
+	struct ma35_crtc *crtc;
+	struct ma35_interface *interface;
+	struct clk *dcuclk;
+	struct clk *dcupclk;
+};
+
+#endif
diff --git a/drivers/gpu/drm/nuvoton/ma35_interface.c b/drivers/gpu/drm/nuvoton/ma35_interface.c
new file mode 100644
index 000000000000..48d1535ace2f
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/ma35_interface.c
@@ -0,0 +1,192 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Nuvoton DRM driver
+ *
+ * Copyright (C) 2026 Nuvoton Technology Corp.
+ *
+ * Author: Joey Lu <a0987203069@...il.com>
+ */
+
+#include <linux/types.h>
+#include <linux/clk.h>
+
+#include <drm/drm_atomic_helper.h>
+#include <drm/drm_bridge.h>
+#include <drm/drm_connector.h>
+#include <drm/drm_drv.h>
+#include <drm/drm_edid.h>
+#include <drm/drm_encoder.h>
+#include <drm/drm_gem_dma_helper.h>
+#include <drm/drm_modeset_helper_vtables.h>
+#include <drm/drm_of.h>
+#include <drm/drm_panel.h>
+#include <drm/drm_print.h>
+#include <drm/drm_probe_helper.h>
+#include <drm/drm_simple_kms_helper.h>
+
+#include "ma35_drm.h"
+
+#define ma35_encoder(e) \
+	container_of(e, struct ma35_interface, drm_encoder)
+#define ma35_connector(c) \
+	container_of(c, struct ma35_interface, drm_connector)
+
+static void ma35_encoder_mode_set(struct drm_encoder *encoder,
+	struct drm_crtc_state *crtc_state,
+	struct drm_connector_state *conn_state)
+{
+	struct drm_device *drm_dev = encoder->dev;
+	struct ma35_drm *priv = ma35_drm(drm_dev);
+	struct drm_display_mode *adjusted_mode = &crtc_state->adjusted_mode;
+	int result;
+
+	clk_set_rate(priv->dcupclk, adjusted_mode->clock * 1000);
+	result = DIV_ROUND_UP(clk_get_rate(priv->dcupclk), 1000);
+	drm_dbg(drm_dev, "Pixel clock: %d kHz; request : %d kHz\n", result, adjusted_mode->clock);
+}
+
+static int ma35_encoder_atomic_check(struct drm_encoder *encoder,
+					struct drm_crtc_state *crtc_state,
+					struct drm_connector_state *conn_state)
+{
+	struct ma35_interface *interface = ma35_encoder(encoder);
+	struct drm_display_info *display_info = &conn_state->connector->display_info;
+
+	interface->bus_flags = display_info->bus_flags;
+
+	return 0;
+}
+
+static const struct drm_encoder_helper_funcs ma35_encoder_helper_funcs = {
+	.atomic_mode_set	= ma35_encoder_mode_set,
+	.atomic_check		= ma35_encoder_atomic_check,
+};
+
+static const struct drm_connector_funcs ma35_connector_funcs = {
+	.reset			= drm_atomic_helper_connector_reset,
+	.fill_modes		= drm_helper_probe_single_connector_modes,
+	.destroy		= drm_connector_cleanup,
+	.atomic_duplicate_state	= drm_atomic_helper_connector_duplicate_state,
+	.atomic_destroy_state	= drm_atomic_helper_connector_destroy_state,
+};
+
+static int ma35_connector_get_modes(struct drm_connector *drm_connector)
+{
+	struct ma35_drm *priv = ma35_drm(drm_connector->dev);
+	struct drm_device *drm_dev = &priv->drm_dev;
+	struct drm_mode_config *mode_config = &drm_dev->mode_config;
+	struct ma35_interface *interface = ma35_connector(drm_connector);
+	int count;
+
+	if (!interface->drm_panel) {
+		/* Use the default modes */
+		count = drm_add_modes_noedid(drm_connector,
+				mode_config->max_width, mode_config->max_height);
+		drm_set_preferred_mode(drm_connector,
+				mode_config->max_width, mode_config->max_height);
+
+		return count;
+	} else {
+		return drm_panel_get_modes(interface->drm_panel, drm_connector);
+	}
+}
+
+static const struct drm_connector_helper_funcs ma35_connector_helper_funcs = {
+	.get_modes		= ma35_connector_get_modes,
+};
+
+static void ma35_encoder_attach_crtc(struct ma35_drm *priv)
+{
+	uint32_t possible_crtcs = drm_crtc_mask(&priv->crtc->drm_crtc);
+
+	priv->interface->drm_encoder.possible_crtcs = possible_crtcs;
+}
+
+static int ma35_bridge_try_attach(struct ma35_drm *priv, struct ma35_interface *interface)
+{
+	struct drm_device *drm_dev = &priv->drm_dev;
+	struct device *dev = drm_dev->dev;
+	struct device_node *of_node = dev->of_node;
+	struct drm_bridge *bridge;
+	struct drm_panel *panel;
+	int ret;
+
+	ret = drm_of_find_panel_or_bridge(of_node, 0, 0, &panel, &bridge);
+
+	if (ret) {
+		drm_info(drm_dev, "No panel or bridge found\n");
+		return ret;
+	}
+
+	if (panel) {
+		bridge = drm_panel_bridge_add_typed(panel, DRM_MODE_CONNECTOR_DPI);
+		if (IS_ERR(bridge))
+			return PTR_ERR(bridge);
+	}
+
+	interface->drm_panel = panel;
+	interface->drm_bridge = bridge;
+
+	ret = drm_bridge_attach(&interface->drm_encoder, bridge,
+				NULL, 0);
+	if (ret) {
+		drm_err(drm_dev, "Failed to attach bridge to encoder\n");
+		return ret;
+	}
+
+	return 0;
+}
+
+int ma35_interface_init(struct ma35_drm *priv)
+{
+	struct ma35_interface *interface;
+	struct drm_device *drm_dev = &priv->drm_dev;
+	struct drm_encoder *drm_encoder;
+	int ret;
+
+	/* encoder */
+	interface = drmm_simple_encoder_alloc(drm_dev,
+					struct ma35_interface, drm_encoder, DRM_MODE_ENCODER_DPI);
+	if (!interface) {
+		drm_err(drm_dev, "Failed to initialize encoder\n");
+		goto error_early;
+	}
+	priv->interface = interface;
+	drm_encoder = &interface->drm_encoder;
+	drm_encoder_helper_add(drm_encoder,
+			       &ma35_encoder_helper_funcs);
+
+	/* attach encoder to crtc */
+	ma35_encoder_attach_crtc(priv);
+
+	/* attach bridge to encoder if found one in device tree */
+	ret = ma35_bridge_try_attach(priv, interface);
+	if (!ret)
+		return 0;
+
+	/* fallback to raw dpi connector */
+	ret = drm_connector_init(drm_dev, &interface->drm_connector,
+					&ma35_connector_funcs,
+					DRM_MODE_CONNECTOR_DPI);
+	if (ret) {
+		drm_err(drm_dev, "Failed to initialize connector\n");
+		goto error_encoder;
+	}
+	drm_connector_helper_add(&interface->drm_connector,
+						&ma35_connector_helper_funcs);
+	ret = drm_connector_attach_encoder(&interface->drm_connector,
+						drm_encoder);
+	if (ret) {
+		drm_err(drm_dev,
+			"Failed to attach connector to encoder\n");
+		goto error_encoder;
+	}
+
+	return ret;
+
+error_encoder:
+	drm_encoder_cleanup(drm_encoder);
+
+error_early:
+	return ret;
+}
diff --git a/drivers/gpu/drm/nuvoton/ma35_interface.h b/drivers/gpu/drm/nuvoton/ma35_interface.h
new file mode 100644
index 000000000000..db7ed41bee45
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/ma35_interface.h
@@ -0,0 +1,30 @@
+/* SPDX-License-Identifier: GPL-2.0+ */
+/*
+ * Nuvoton DRM driver
+ *
+ * Copyright (C) 2026 Nuvoton Technology Corp.
+ *
+ * Author: Joey Lu <a0987203069@...il.com>
+ */
+
+#ifndef _MA35_INTERFACE_H_
+#define _MA35_INTERFACE_H_
+
+#include <drm/drm_bridge.h>
+#include <drm/drm_connector.h>
+#include <drm/drm_encoder.h>
+
+struct ma35_drm;
+
+struct ma35_interface {
+	struct drm_encoder drm_encoder;
+	struct drm_connector drm_connector;
+	struct drm_panel *drm_panel;
+	struct drm_bridge *drm_bridge;
+
+	u32 bus_flags;
+};
+
+int ma35_interface_init(struct ma35_drm *priv);
+
+#endif
diff --git a/drivers/gpu/drm/nuvoton/ma35_plane.c b/drivers/gpu/drm/nuvoton/ma35_plane.c
new file mode 100644
index 000000000000..3449cd33059a
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/ma35_plane.c
@@ -0,0 +1,904 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Nuvoton DRM driver
+ *
+ * Copyright (C) 2026 Nuvoton Technology Corp.
+ *
+ * Author: Joey Lu <a0987203069@...il.com>
+ */
+
+#include <linux/of.h>
+#include <linux/types.h>
+
+#include <drm/drm_atomic.h>
+#include <drm/drm_atomic_helper.h>
+#include <drm/drm_blend.h>
+#include <drm/drm_fb_dma_helper.h>
+#include <drm/drm_fourcc.h>
+#include <drm/drm_framebuffer.h>
+#include <drm/drm_gem_atomic_helper.h>
+#include <drm/drm_plane.h>
+#include <drm/drm_print.h>
+
+#include "ma35_drm.h"
+
+#define ma35_layer(p) \
+	container_of(p, struct ma35_layer, drm_plane)
+
+static uint32_t ma35_layer_formats[] = {
+	/* rgb32 */
+	DRM_FORMAT_ARGB8888,
+	DRM_FORMAT_XRGB8888,
+	DRM_FORMAT_ARGB2101010,
+	/* rgb16 */
+	DRM_FORMAT_XRGB4444,
+	DRM_FORMAT_ARGB4444,
+	DRM_FORMAT_XRGB1555,
+	DRM_FORMAT_ARGB1555,
+	DRM_FORMAT_RGB565,
+	/* yuv */
+	DRM_FORMAT_YUYV,
+	DRM_FORMAT_UYVY,
+	DRM_FORMAT_YVU420,
+	DRM_FORMAT_NV12,
+	DRM_FORMAT_NV16,
+	DRM_FORMAT_P010,
+};
+
+static uint32_t ma35_cursor_formats[] = {
+	DRM_FORMAT_XRGB8888,
+};
+
+static struct ma35_plane_property ma35_plane_properties[] = {
+	{ /* overlay */
+		.fb_addr    = { MA35_OVERLAY_ADDRESS,
+						MA35_OVERLAY_UPLANAR_ADDRESS,
+						MA35_OVERLAY_VPLANAR_ADDRESS },
+		.fb_stride  = { MA35_OVERLAY_STRIDE,
+						MA35_OVERLAY_USTRIDE,
+						MA35_OVERLAY_VSTRIDE },
+		.alpha      = true,
+		.swizzle    = true,
+		.colorkey   = true, // ARGB only, replaced with primary
+		.background = false,
+		.foreground = false,
+	},
+	{ /* primary */
+		.fb_addr    = { MA35_FRAMEBUFFER_ADDRESS,
+						MA35_FRAMEBUFFER_UPLANAR_ADDRESS,
+						MA35_FRAMEBUFFER_VPLANAR_ADDRESS },
+		.fb_stride  = { MA35_FRAMEBUFFER_STRIDE,
+						MA35_FRAMEBUFFER_USTRIDE,
+						MA35_FRAMEBUFFER_VSTRIDE },
+		.alpha      = false,
+		.swizzle    = true,
+		.colorkey   = true, // ARGB only, replaced with background
+		.background = true,
+		.foreground = false,
+	},
+	{ /* cursor */
+		.alpha      = false,
+		.swizzle    = false,
+		.colorkey   = false,
+		.background = true,
+		.foreground = true,
+	},
+};
+
+static const struct drm_prop_enum_list ma35_blend_modes[] = {
+	{ MA35_ALPHA_CLEAR, "CLEAR" },
+	{ MA35_ALPHA_SRC, "SRC" },
+	{ MA35_ALPHA_DST, "DST" },
+	{ MA35_ALPHA_SRC_OVER, "SRC_OVER" },
+	{ MA35_ALPHA_DST_OVER, "DST_OVER" },
+	{ MA35_ALPHA_SRC_IN, "SRC_IN" },
+	{ MA35_ALPHA_DST_IN, "DST_IN" },
+	{ MA35_ALPHA_SRC_OUT, "SRC_OUT" },
+	{ MA35_ALPHA_DST_OUT, "DST_OUT" },
+	{ MA35_ALPHA_SRC_ATOP, "SRC_ATOP" },
+	{ MA35_ALPHA_DST_ATOP, "DST_ATOP" },
+	{ MA35_ALPHA_XOR, "XOR" },
+};
+
+static const struct drm_prop_enum_list ma35_alpha_modes[] = {
+	{ MA35_ALPHA_MODE_NONE, "None" },
+	{ MA35_ALPHA_MODE_GLOBAL, "Coverage" },
+};
+
+static const struct drm_prop_enum_list ma35_swizzles[] = {
+	{ MA35_SWIZZLE_ARGB, "ARGB" },
+	{ MA35_SWIZZLE_RGBA, "RGBA" },
+	{ MA35_SWIZZLE_ABGR, "ABGR" },
+	{ MA35_SWIZZLE_BGRA, "BGRA" },
+	{ MA35_SWIZZLE_UV, "UV" },
+};
+
+static int ma35_layer_format_validate(u32 fourcc, u32 *format)
+{
+	switch (fourcc) {
+	case DRM_FORMAT_XRGB4444:
+		*format = MA35_FORMAT_X4R4G4B4;
+		break;
+	case DRM_FORMAT_ARGB4444:
+		*format = MA35_FORMAT_A4R4G4B4;
+		break;
+	case DRM_FORMAT_XRGB1555:
+		*format = MA35_FORMAT_X1R5G5B5;
+		break;
+	case DRM_FORMAT_ARGB1555:
+		*format = MA35_FORMAT_A1R5G5B5;
+		break;
+	case DRM_FORMAT_RGB565:
+		*format = MA35_FORMAT_R5G6B5;
+		break;
+	case DRM_FORMAT_XRGB8888:
+		*format = MA35_FORMAT_X8R8G8B8;
+		break;
+	case DRM_FORMAT_ARGB8888:
+		*format = MA35_FORMAT_A8R8G8B8;
+		break;
+	case DRM_FORMAT_ARGB2101010:
+		*format = MA35_FORMAT_A2R10G10B10;
+		break;
+	case DRM_FORMAT_YUYV:
+		*format = MA35_FORMAT_YUY2;
+		break;
+	case DRM_FORMAT_UYVY:
+		*format = MA35_FORMAT_UYVY;
+		break;
+	case DRM_FORMAT_YVU420:
+		*format = MA35_FORMAT_YV12;
+		break;
+	case DRM_FORMAT_NV12:
+		*format = MA35_FORMAT_NV12;
+		break;
+	case DRM_FORMAT_NV16:
+		*format = MA35_FORMAT_NV16;
+		break;
+	case DRM_FORMAT_P010:
+		*format = MA35_FORMAT_P010;
+		break;
+	default:
+		return -EINVAL;
+	}
+
+	return 0;
+}
+
+static int ma35_layer_blend_mode_select(u32 mode, u32 *reg)
+{
+	u32 ret = 0;
+
+	switch (mode) {
+	case MA35_ALPHA_CLEAR:
+		*reg = MA35_BLEND_MODE_CLEAR;
+		break;
+	case MA35_ALPHA_SRC:
+		*reg = MA35_BLEND_MODE_SRC;
+		break;
+	case MA35_ALPHA_DST:
+		*reg = MA35_BLEND_MODE_DST;
+		break;
+	case MA35_ALPHA_SRC_OVER:
+		*reg = MA35_BLEND_MODE_SRC_OVER;
+		break;
+	case MA35_ALPHA_DST_OVER:
+		*reg = MA35_BLEND_MODE_DST_OVER;
+		break;
+	case MA35_ALPHA_SRC_IN:
+		*reg = MA35_BLEND_MODE_SRC_IN;
+		break;
+	case MA35_ALPHA_DST_IN:
+		*reg = MA35_BLEND_MODE_DST_IN;
+		break;
+	case MA35_ALPHA_SRC_OUT:
+		*reg = MA35_BLEND_MODE_SRC_OUT;
+		break;
+	case MA35_ALPHA_DST_OUT:
+		*reg = MA35_BLEND_MODE_DST_OUT;
+		break;
+	case MA35_ALPHA_SRC_ATOP:
+		*reg = MA35_BLEND_MODE_SRC_ATOP;
+		break;
+	case MA35_ALPHA_DST_ATOP:
+		*reg = MA35_BLEND_MODE_DST_ATOP;
+		break;
+	case MA35_ALPHA_XOR:
+		*reg = MA35_BLEND_MODE_XOR;
+		break;
+	default:
+		ret = -EINVAL;
+	}
+
+	return ret;
+}
+
+static int ma35_plane_atomic_check(struct drm_plane *drm_plane,
+				      struct drm_atomic_state *state)
+{
+	struct drm_device *drm_dev = drm_plane->dev;
+	struct ma35_layer *layer = ma35_layer(drm_plane);
+	struct drm_plane_state *new_state =
+		drm_atomic_get_new_plane_state(state, drm_plane);
+	struct drm_crtc *crtc = new_state->crtc;
+	struct drm_framebuffer *fb = new_state->fb;
+	struct drm_crtc_state *crtc_state;
+	bool can_position;
+	u32 format;
+
+	if (!crtc)
+		return 0;
+
+	crtc_state = drm_atomic_get_new_crtc_state(state, crtc);
+	if (WARN_ON(!crtc_state))
+		return -EINVAL;
+
+	if (new_state->crtc_x < 0 || new_state->crtc_y < 0) {
+		drm_err(drm_dev,
+			"Negative on-CRTC positions are not supported.\n");
+		return -EINVAL;
+	}
+
+	if (layer->config.blend_mode > MA35_ALPHA_XOR) {
+		drm_err(drm_dev, "Invalid blend mode\n");
+		return -EINVAL;
+	}
+
+	if (layer->config.swizzle > MA35_SWIZZLE_UV) {
+		drm_err(drm_dev, "Invalid swizzle mode\n");
+		return -EINVAL;
+	}
+
+	if ((layer->config.swizzle & MA35_SWIZZLE_ARGB_MASK) &&
+		fb->format->is_yuv) {
+		drm_err(drm_dev, "Invalid swizzle mode for RGB format\n");
+		return -EINVAL;
+	}
+
+	if ((layer->config.swizzle & MA35_SWIZZLE_UV_MASK) &&
+		!fb->format->is_yuv) {
+		drm_err(drm_dev, "Invalid swizzle mode for YUV format\n");
+		return -EINVAL;
+	}
+
+	if (ma35_layer_format_validate(fb->format->format, &format) < 0) {
+		drm_err(drm_dev, "Unsupported format\n");
+		return -EINVAL;
+	}
+
+	can_position = (drm_plane->type != DRM_PLANE_TYPE_PRIMARY);
+	return drm_atomic_helper_check_plane_state(new_state, crtc_state,
+						  DRM_PLANE_NO_SCALING, DRM_PLANE_NO_SCALING,
+						  can_position, true);
+}
+
+static int ma35_cursor_plane_atomic_check(struct drm_plane *drm_plane,
+				      struct drm_atomic_state *state)
+{
+	struct drm_device *drm_dev = drm_plane->dev;
+	struct drm_plane_state *new_state =
+		drm_atomic_get_new_plane_state(state, drm_plane);
+	struct drm_framebuffer *fb = new_state->fb;
+	struct drm_crtc *crtc = new_state->crtc;
+	struct drm_crtc_state *crtc_state;
+
+	if (!fb)
+		return 0;
+
+	if (!crtc)
+		return -EINVAL;
+
+	crtc_state = drm_atomic_get_new_crtc_state(state, crtc);
+	if (WARN_ON(!crtc_state))
+		return -EINVAL;
+
+	if (fb->format->format != DRM_FORMAT_XRGB8888) {
+		drm_err(drm_dev, "Invalid cursor format\n");
+		return -EINVAL;
+	}
+
+	if (new_state->crtc_w != MA35_CURSOR_SIZE || new_state->crtc_h != MA35_CURSOR_SIZE) {
+		drm_err(drm_dev, "Unsupported cursor size: %ux%u\n",
+				new_state->crtc_w, new_state->crtc_h);
+		return -EINVAL;
+	}
+
+	if (new_state->hotspot_x >= 32 || new_state->hotspot_x < 0 ||
+		new_state->hotspot_y >= 32 || new_state->hotspot_y < 0) {
+		drm_err(drm_dev, "Invalid cursor hotspot offset\n");
+		return -EINVAL;
+	}
+
+	return drm_atomic_helper_check_plane_state(new_state, crtc_state,
+						  DRM_PLANE_NO_SCALING, DRM_PLANE_NO_SCALING,
+						  true, true);
+}
+
+static int ma35_cursor_plane_atomic_async_check(struct drm_plane *drm_plane,
+				      struct drm_atomic_state *state, bool flip)
+{
+	return ma35_cursor_plane_atomic_check(drm_plane, state);
+}
+
+static void ma35_overlay_position_update(struct ma35_drm *priv,
+						int x, int y, uint32_t w, uint32_t h)
+{
+	u32 reg;
+	int right, bottom;
+
+	right = x + w;
+	bottom = y + h;
+
+	x = (x < 0) ? 0 : x;
+	y = (y < 0) ? 0 : y;
+	right = (right < 0) ? 0 : right;
+	bottom = (bottom < 0) ? 0 : bottom;
+
+	reg = FIELD_PREP(MA35_OVERLAY_POSITION_X_MASK, x) |
+		  FIELD_PREP(MA35_OVERLAY_POSITION_Y_MASK, y);
+	regmap_write(priv->regmap, MA35_OVERLAY_TL, reg);
+
+	reg = FIELD_PREP(MA35_OVERLAY_POSITION_X_MASK, right) |
+		  FIELD_PREP(MA35_OVERLAY_POSITION_Y_MASK, bottom);
+	regmap_write(priv->regmap, MA35_OVERLAY_BR, reg);
+}
+
+static void ma35_plane_atomic_update(struct drm_plane *drm_plane,
+					struct drm_atomic_state *state)
+{
+	struct ma35_layer *layer = ma35_layer(drm_plane);
+	struct ma35_drm *priv = ma35_drm(drm_plane->dev);
+	struct drm_plane_state *new_state =
+		drm_atomic_get_new_plane_state(state, drm_plane);
+	struct drm_framebuffer *fb = new_state->fb;
+	u32 format, reg;
+	u32 *preg;
+
+	ma35_layer_format_validate(fb->format->format, &format);
+
+	if (drm_plane->type == DRM_PLANE_TYPE_PRIMARY) {
+		reg = FIELD_PREP(MA35_PRIMARY_FORMAT_MASK, format) |
+			  FIELD_PREP(MA35_PRIMARY_SWIZZLE_MASK, layer->config.swizzle) |
+			  MA35_PRIMARY_RESET | MA35_PRIMARY_ENABLE;
+		if (layer->config.colorkeylo || layer->config.colorkeyup)
+			reg |= FIELD_PREP(MA35_PRIMARY_TRANSPARENCY_MASK, MA35_COLORKEY_ENABLE);
+		else
+			reg |= FIELD_PREP(MA35_PRIMARY_TRANSPARENCY_MASK, MA35_COLORKEY_DISABLE);
+
+		regmap_write(priv->regmap, MA35_FRAMEBUFFER_CONFIG, reg);
+
+		reg = FIELD_PREP(MA35_LAYER_FB_HEIGHT, fb->height) |
+			  FIELD_PREP(MA35_LAYER_FB_WIDTH, fb->width);
+		regmap_write(priv->regmap, MA35_FRAMEBUFFER_SIZE, reg);
+
+		/* background */
+		regmap_write(priv->regmap, MA35_FRAMEBUFFER_BGCOLOR, layer->config.background);
+
+		/* clear value */
+		regmap_write(priv->regmap, MA35_FRAMEBUFFER_CLEARVALUE, 0);
+
+		/* colorkey */
+		regmap_write(priv->regmap, MA35_FRAMEBUFFER_COLORKEY, layer->config.colorkeylo);
+		regmap_write(priv->regmap, MA35_FRAMEBUFFER_COLORHIGHKEY, layer->config.colorkeyup);
+	} else if (drm_plane->type == DRM_PLANE_TYPE_OVERLAY) {
+		reg = FIELD_PREP(MA35_OVERLAY_FORMAT_MASK, format) |
+			  FIELD_PREP(MA35_OVERLAY_SWIZZLE_MASK, layer->config.swizzle) |
+			  MA35_OVERLAY_ENABLE;
+		if (layer->config.colorkeylo || layer->config.colorkeyup)
+			reg |= FIELD_PREP(MA35_OVERLAY_TRANSPARENCY_MASK, MA35_COLORKEY_ENABLE);
+		else
+			reg |= FIELD_PREP(MA35_OVERLAY_TRANSPARENCY_MASK, MA35_COLORKEY_DISABLE);
+
+		regmap_write(priv->regmap, MA35_OVERLAY_CONFIG, reg);
+
+		reg = FIELD_PREP(MA35_LAYER_FB_HEIGHT, fb->height) |
+			FIELD_PREP(MA35_LAYER_FB_WIDTH, fb->width);
+		regmap_write(priv->regmap, MA35_OVERLAY_SIZE, reg);
+		/* can_position */
+		ma35_overlay_position_update(priv, new_state->crtc_x, new_state->crtc_y,
+			new_state->crtc_w, new_state->crtc_h);
+		/* alpha blending */
+		if (fb->format->format == DRM_FORMAT_ARGB8888) {
+			ma35_layer_blend_mode_select(layer->config.blend_mode, &reg);
+			reg |= FIELD_PREP(MA35_SRC_ALPHA_MODE, (u32)layer->config.alpha_mode[0]) |
+				FIELD_PREP(MA35_DST_ALPHA_MODE, (u32)layer->config.alpha_mode[1]);
+			regmap_write(priv->regmap, MA35_OVERLAY_ALPHA_BLEND_CONFIG, reg);
+
+			regmap_write(priv->regmap, MA35_OVERLAY_SRC_GLOBAL_COLOR,
+						 layer->config.src_color);
+			regmap_write(priv->regmap, MA35_OVERLAY_DST_GLOBAL_COLOR,
+						 layer->config.dst_color);
+		} else {
+			regmap_update_bits(priv->regmap, MA35_OVERLAY_ALPHA_BLEND_CONFIG,
+				MA35_ALPHA_BLEND_DISABLE, MA35_ALPHA_BLEND_DISABLE);
+		}
+
+		/* clear value */
+		regmap_write(priv->regmap, MA35_OVERLAY_CLEAR_VALUE, 0);
+
+		/* colorkey */
+		regmap_write(priv->regmap, MA35_OVERLAY_COLOR_KEY, layer->config.colorkeylo);
+		regmap_write(priv->regmap, MA35_OVERLAY_COLOR_KEY_HIGH, layer->config.colorkeyup);
+	}
+
+	/* retrieves DMA address set by userspace */
+	for (int i = 0; i < fb->format->num_planes; i++) {
+		layer->fb_base[i] = drm_fb_dma_get_gem_addr(fb, new_state, i);
+		preg = ma35_plane_properties[drm_plane->type].fb_addr;
+		regmap_write(priv->regmap, preg[i], layer->fb_base[i]);
+		preg = ma35_plane_properties[drm_plane->type].fb_stride;
+		regmap_write(priv->regmap, preg[i], fb->pitches[i]);
+	}
+}
+
+static void ma35_cursor_position_update(struct ma35_drm *priv, int x, int y)
+{
+	u32 reg;
+
+	x = (x < 0) ? 0 : x;
+	y = (y < 0) ? 0 : y;
+
+	reg = FIELD_PREP(MA35_CURSOR_X_MASK, x) |
+		  FIELD_PREP(MA35_CURSOR_Y_MASK, y);
+	regmap_write(priv->regmap, MA35_CURSOR_LOCATION, reg);
+}
+
+static void ma35_cursor_plane_atomic_update(struct drm_plane *drm_plane,
+					struct drm_atomic_state *state)
+{
+	struct ma35_layer *layer = ma35_layer(drm_plane);
+	struct ma35_drm *priv = ma35_drm(drm_plane->dev);
+	struct drm_plane_state *old_state =
+		drm_atomic_get_old_plane_state(state, drm_plane);
+	struct drm_plane_state *new_state =
+		drm_atomic_get_new_plane_state(state, drm_plane);
+	struct drm_framebuffer *old_fb = old_state->fb;
+	struct drm_framebuffer *new_fb = new_state->fb;
+	u32 reg;
+
+	if (!new_state->visible) {
+		regmap_update_bits(priv->regmap, MA35_CURSOR_CONFIG,
+			MA35_CURSOR_FORMAT_MASK, MA35_CURSOR_FORMAT_DISABLE);
+		return;
+	}
+
+	/* update position */
+	ma35_cursor_position_update(priv, new_state->crtc_x, new_state->crtc_y);
+
+	/* check new_state is different from old_state for dimensions or format changed */
+	if (!old_fb || old_fb != new_fb) {
+		layer->fb_base[0] = drm_fb_dma_get_gem_addr(new_fb, new_state, 0);
+		regmap_write(priv->regmap, MA35_CURSOR_ADDRESS, layer->fb_base[0]);
+
+		regmap_write(priv->regmap, MA35_CURSOR_BACKGROUND, layer->config.background);
+		regmap_write(priv->regmap, MA35_CURSOR_FOREGROUND, layer->config.foreground);
+		regmap_update_bits(priv->regmap, MA35_CURSOR_CONFIG,
+			MA35_CURSOR_FORMAT_MASK, MA35_CURSOR_FORMAT_A8R8G8B8);
+	}
+
+	/* update hotspot offset & format */
+	if (old_state->hotspot_x != new_state->hotspot_x ||
+		old_state->hotspot_y != new_state->hotspot_y) {
+		reg = MA35_CURSOR_FORMAT_A8R8G8B8 |
+			FIELD_PREP(MA35_CURSOR_HOTSPOT_X_MASK, new_state->hotspot_x) |
+			FIELD_PREP(MA35_CURSOR_HOTSPOT_Y_MASK, new_state->hotspot_y);
+		regmap_write(priv->regmap, MA35_CURSOR_CONFIG, reg);
+	}
+}
+
+static void ma35_cursor_plane_atomic_async_update(struct drm_plane *drm_plane,
+					struct drm_atomic_state *state)
+{
+	struct ma35_layer *layer = ma35_layer(drm_plane);
+	struct ma35_drm *priv = ma35_drm(drm_plane->dev);
+	struct drm_plane_state *old_state = drm_plane->state;
+	struct drm_plane_state *new_state =
+		drm_atomic_get_new_plane_state(state, drm_plane);
+	struct drm_framebuffer *old_fb = old_state->fb;
+	struct drm_framebuffer *new_fb = new_state->fb;
+	u32 reg;
+
+	/* update the current one with the new plane state */
+	old_state->crtc_x = new_state->crtc_x;
+	old_state->crtc_y = new_state->crtc_y;
+	old_state->crtc_h = new_state->crtc_h;
+	old_state->crtc_w = new_state->crtc_w;
+	old_state->src_x = new_state->src_x;
+	old_state->src_y = new_state->src_y;
+	old_state->src_h = new_state->src_h;
+	old_state->src_w = new_state->src_w;
+	/* swap current and new framebuffers */
+	swap(old_fb, new_fb);
+
+	if (!new_state->visible) {
+		regmap_update_bits(priv->regmap, MA35_CURSOR_CONFIG,
+			MA35_CURSOR_FORMAT_MASK, MA35_CURSOR_FORMAT_DISABLE);
+		return;
+	}
+
+	/* update position */
+	ma35_cursor_position_update(priv, new_state->crtc_x, new_state->crtc_y);
+
+	/* check new_state is different from old_state for dimensions or format changed */
+	if (!old_fb || old_fb != new_fb) {
+		layer->fb_base[0] = drm_fb_dma_get_gem_addr(new_fb, new_state, 0);
+		regmap_write(priv->regmap, MA35_CURSOR_ADDRESS, layer->fb_base[0]);
+
+		regmap_write(priv->regmap, MA35_CURSOR_BACKGROUND, layer->config.background);
+		regmap_write(priv->regmap, MA35_CURSOR_FOREGROUND, layer->config.foreground);
+		regmap_update_bits(priv->regmap, MA35_CURSOR_CONFIG,
+			MA35_CURSOR_FORMAT_MASK, MA35_CURSOR_FORMAT_A8R8G8B8);
+	}
+
+	/* update hotspot offset & format */
+	if (old_state->hotspot_x != new_state->hotspot_x ||
+		old_state->hotspot_y != new_state->hotspot_y) {
+		reg = MA35_CURSOR_FORMAT_A8R8G8B8 |
+			FIELD_PREP(MA35_CURSOR_HOTSPOT_X_MASK, new_state->hotspot_x) |
+			FIELD_PREP(MA35_CURSOR_HOTSPOT_Y_MASK, new_state->hotspot_y);
+		regmap_write(priv->regmap, MA35_CURSOR_CONFIG, reg);
+		old_state->hotspot_x = new_state->hotspot_x;
+		old_state->hotspot_y = new_state->hotspot_y;
+	}
+}
+
+static void ma35_plane_atomic_disable(struct drm_plane *drm_plane,
+					 struct drm_atomic_state *state)
+{
+	struct ma35_drm *priv = ma35_drm(drm_plane->dev);
+
+	regmap_update_bits(priv->regmap, MA35_FRAMEBUFFER_CONFIG,
+		MA35_PRIMARY_ENABLE, 0);
+}
+
+static void ma35_cursor_plane_atomic_disable(struct drm_plane *drm_plane,
+					 struct drm_atomic_state *state)
+{
+	struct ma35_drm *priv = ma35_drm(drm_plane->dev);
+
+	regmap_update_bits(priv->regmap, MA35_CURSOR_CONFIG,
+		MA35_CURSOR_FORMAT_MASK, MA35_CURSOR_FORMAT_DISABLE);
+}
+
+static struct drm_plane_helper_funcs ma35_plane_helper_funcs = {
+	.atomic_check		= ma35_plane_atomic_check,
+	.atomic_update		= ma35_plane_atomic_update,
+	.atomic_disable		= ma35_plane_atomic_disable,
+};
+
+static struct drm_plane_helper_funcs ma35_cursor_plane_helper_funcs = {
+	.atomic_check		= ma35_cursor_plane_atomic_check,
+	.atomic_update		= ma35_cursor_plane_atomic_update,
+	.atomic_disable		= ma35_cursor_plane_atomic_disable,
+	.atomic_async_check		= ma35_cursor_plane_atomic_async_check,
+	.atomic_async_update	= ma35_cursor_plane_atomic_async_update,
+};
+
+static int ma35_plane_set_property(struct drm_plane *drm_plane,
+	struct drm_plane_state *state, struct drm_property *property,
+	uint64_t val)
+{
+	struct ma35_layer *layer = ma35_layer(drm_plane);
+
+	if (property == layer->blend_mode_prop)
+		layer->config.blend_mode = val;
+	else if (property == layer->src_alpha_mode_prop)
+		layer->config.alpha_mode[0] = val;
+	else if (property == layer->dst_alpha_mode_prop)
+		layer->config.alpha_mode[1] = val;
+	else if (property == layer->src_color_prop)
+		layer->config.src_color = val;
+	else if (property == layer->dst_color_prop)
+		layer->config.dst_color = val;
+	else if (property == layer->swizzle_prop)
+		layer->config.swizzle = val;
+	else if (property == layer->colorkeylo_prop)
+		layer->config.colorkeylo = val;
+	else if (property == layer->colorkeyup_prop)
+		layer->config.colorkeyup = val;
+	else if (property == layer->background_prop)
+		layer->config.background = val;
+	else if (property == layer->foreground_prop)
+		layer->config.foreground = val;
+	else if (property == drm_plane->hotspot_x_property)
+		state->hotspot_x = val;
+	else if (property == drm_plane->hotspot_y_property)
+		state->hotspot_y = val;
+	else
+		return -EINVAL;
+
+	return 0;
+}
+
+static int ma35_plane_get_property(struct drm_plane *drm_plane,
+	const struct drm_plane_state *state, struct drm_property *property,
+	uint64_t *val)
+{
+	struct ma35_layer *layer = ma35_layer(drm_plane);
+
+	if (property == layer->blend_mode_prop)
+		*val = layer->config.blend_mode;
+	else if (property == layer->src_alpha_mode_prop)
+		*val = layer->config.alpha_mode[0];
+	else if (property == layer->dst_alpha_mode_prop)
+		*val = layer->config.alpha_mode[1];
+	else if (property == layer->src_color_prop)
+		*val = layer->config.src_color;
+	else if (property == layer->dst_color_prop)
+		*val = layer->config.dst_color;
+	else if (property == layer->swizzle_prop)
+		*val = layer->config.swizzle;
+	else if (property == layer->colorkeylo_prop)
+		*val = layer->config.colorkeylo;
+	else if (property == layer->colorkeyup_prop)
+		*val = layer->config.colorkeyup;
+	else if (property == layer->background_prop)
+		*val = layer->config.background;
+	else if (property == layer->foreground_prop)
+		*val = layer->config.foreground;
+	else if (property == drm_plane->hotspot_x_property)
+		*val = state->hotspot_x;
+	else if (property == drm_plane->hotspot_y_property)
+		*val = state->hotspot_y;
+	else
+		return -EINVAL;
+
+	return 0;
+}
+
+static const struct drm_plane_funcs ma35_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,
+	.atomic_set_property = ma35_plane_set_property,
+	.atomic_get_property = ma35_plane_get_property,
+};
+
+static int ma35_layer_create_properties(struct ma35_drm *priv,
+				      struct ma35_layer *layer)
+{
+	struct drm_device *drm_dev = &priv->drm_dev;
+	struct drm_plane *drm_plane = &layer->drm_plane;
+	int ret = 0;
+
+	if (ma35_plane_properties[drm_plane->type].alpha) {
+		layer->blend_mode_prop = drm_property_create_enum(drm_dev, 0,
+							"porter-duff-blend-mode",
+							ma35_blend_modes,
+							ARRAY_SIZE(ma35_blend_modes));
+		if (!layer->blend_mode_prop) {
+			drm_err(drm_dev, "Failed to create blend mode property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->blend_mode_prop,
+								   MA35_ALPHA_SRC_OVER);
+		layer->config.blend_mode = MA35_ALPHA_SRC_OVER;
+
+		layer->src_alpha_mode_prop = drm_property_create_enum(drm_dev, 0,
+							"source-alpha-mode",
+							ma35_alpha_modes,
+							ARRAY_SIZE(ma35_alpha_modes));
+		if (!layer->src_alpha_mode_prop) {
+			drm_err(drm_dev, "Failed to create source alpha property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->src_alpha_mode_prop,
+								   MA35_ALPHA_MODE_NONE);
+		layer->config.alpha_mode[0] = MA35_ALPHA_MODE_NONE;
+
+		layer->dst_alpha_mode_prop = drm_property_create_enum(drm_dev, 0,
+							"destination-alpha-mode",
+							ma35_alpha_modes,
+							ARRAY_SIZE(ma35_alpha_modes));
+		if (!layer->dst_alpha_mode_prop) {
+			drm_err(drm_dev, "Failed to create destination alpha property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->dst_alpha_mode_prop,
+								   MA35_ALPHA_MODE_NONE);
+		layer->config.alpha_mode[1] = MA35_ALPHA_MODE_NONE;
+
+		layer->src_color_prop = drm_property_create_range(drm_dev, 0,
+							"source-global-color",
+							0, 0xffffffff);
+		if (!layer->src_color_prop) {
+			drm_err(drm_dev, "Failed to create source color property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->src_color_prop, 0);
+		layer->config.src_color = 0;
+
+		layer->dst_color_prop = drm_property_create_range(drm_dev, 0,
+							"destination-global-color",
+							0, 0xffffffff);
+		if (!layer->dst_color_prop) {
+			drm_err(drm_dev, "Failed to create destination color property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->dst_color_prop, 0);
+		layer->config.dst_color = 0;
+	}
+
+	if (ma35_plane_properties[drm_plane->type].swizzle) {
+		layer->swizzle_prop = drm_property_create_enum(drm_dev, 0,
+							"swizzle",
+							ma35_swizzles,
+							ARRAY_SIZE(ma35_swizzles));
+		if (!layer->swizzle_prop) {
+			drm_err(drm_dev, "Failed to create swizzle property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->swizzle_prop,
+								   MA35_SWIZZLE_ARGB);
+		layer->config.swizzle = MA35_SWIZZLE_ARGB;
+	}
+
+	if (ma35_plane_properties[drm_plane->type].colorkey) {
+		layer->colorkeylo_prop = drm_property_create_range(drm_dev, 0,
+							"colorkey-lower-bound",
+							0, 0xffffffff);
+		if (!layer->colorkeylo_prop) {
+			drm_err(drm_dev, "Failed to create colorkey property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->colorkeylo_prop, 0);
+		layer->config.colorkeylo = 0;
+
+		layer->colorkeyup_prop = drm_property_create_range(drm_dev, 0,
+							"colorkey-upper-bound",
+							0, 0xffffffff);
+		if (!layer->colorkeyup_prop) {
+			drm_err(drm_dev, "Failed to create colorkey property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->colorkeyup_prop, 0);
+		layer->config.colorkeyup = 0;
+	}
+
+	if (ma35_plane_properties[drm_plane->type].background) {
+		layer->background_prop = drm_property_create_range(drm_dev, 0,
+							"background",
+							0, 0xffffffff);
+		if (!layer->background_prop) {
+			drm_err(drm_dev, "Failed to create background property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->background_prop, 0);
+		layer->config.background = 0;
+	}
+
+	if (ma35_plane_properties[drm_plane->type].foreground) {
+		layer->foreground_prop = drm_property_create_range(drm_dev, 0,
+							"foreground",
+							0, 0xffffffff);
+		if (!layer->foreground_prop) {
+			drm_err(drm_dev, "Failed to create foreground property\n");
+			return -ENOMEM;
+		}
+		drm_object_attach_property(&drm_plane->base, layer->foreground_prop,
+							0xffffff);
+		layer->config.foreground = 0xffffff;
+	}
+
+	return ret;
+}
+
+struct ma35_layer *ma35_layer_get_from_type(struct ma35_drm *priv, enum drm_plane_type type)
+{
+	struct ma35_layer *layer;
+	struct drm_plane *drm_plane;
+
+	list_for_each_entry(layer, &priv->layers_list, list) {
+		drm_plane = &layer->drm_plane;
+		if (drm_plane->type == type)
+			return layer;
+	}
+
+	return NULL;
+}
+
+static int ma35_layer_create(struct ma35_drm *priv,
+			      struct device_node *of_node, u32 index,
+			      enum drm_plane_type type)
+{
+	struct drm_device *drm_dev = &priv->drm_dev;
+	struct device *dev = drm_dev->dev;
+	struct ma35_layer *layer;
+	int ret;
+
+	layer = devm_kzalloc(dev, sizeof(*layer), GFP_KERNEL);
+	if (!layer) {
+		ret = -ENOMEM;
+		goto error;
+	}
+
+	layer->of_node = of_node;
+
+	if (type == DRM_PLANE_TYPE_CURSOR) {
+		ret = drm_universal_plane_init(drm_dev, &layer->drm_plane,
+			1 << MA35_DEFAULT_CRTC_ID,
+			&ma35_plane_funcs, ma35_cursor_formats,
+			ARRAY_SIZE(ma35_cursor_formats), NULL, type, NULL);
+		if (ret) {
+			drm_err(drm_dev, "Failed to initialize layer plane\n");
+			return ret;
+		}
+
+		drm_plane_helper_add(&layer->drm_plane, &ma35_cursor_plane_helper_funcs);
+	} else {
+		ret = drm_universal_plane_init(drm_dev, &layer->drm_plane,
+			1 << MA35_DEFAULT_CRTC_ID,
+			&ma35_plane_funcs, ma35_layer_formats,
+			ARRAY_SIZE(ma35_layer_formats), NULL, type, NULL);
+		if (ret) {
+			drm_err(drm_dev, "Failed to initialize layer plane\n");
+			return ret;
+		}
+
+		drm_plane_helper_add(&layer->drm_plane, &ma35_plane_helper_funcs);
+	}
+
+	if (ma35_layer_create_properties(priv, layer)) {
+		drm_err(drm_dev, "Failed to parse config for layer #%d\n",
+			index);
+		goto error;
+	}
+
+	drm_plane_create_zpos_immutable_property(&layer->drm_plane, index);
+
+	list_add_tail(&layer->list, &priv->layers_list);
+
+	return 0;
+
+error:
+	if (layer) {
+		list_del(&layer->list);
+		devm_kfree(dev, layer);
+	}
+
+	return ret;
+}
+
+void ma35_overlay_attach_crtc(struct ma35_drm *priv)
+{
+	uint32_t possible_crtcs = drm_crtc_mask(&priv->crtc->drm_crtc);
+	struct ma35_layer *layer;
+	struct drm_plane *drm_plane;
+
+	list_for_each_entry(layer, &priv->layers_list, list) {
+		drm_plane = &layer->drm_plane;
+		if (drm_plane->type != DRM_PLANE_TYPE_OVERLAY)
+			continue;
+
+		drm_plane->possible_crtcs = possible_crtcs;
+	}
+}
+
+int ma35_plane_init(struct ma35_drm *priv)
+{
+	struct drm_device *drm_dev = &priv->drm_dev;
+	int ret;
+
+	ret = ma35_layer_create(priv, NULL, 0, DRM_PLANE_TYPE_PRIMARY);
+	if (ret) {
+		drm_err(drm_dev, "Failed to create primary layer\n");
+		return ret;
+	}
+
+	ret = ma35_layer_create(priv, NULL, 1, DRM_PLANE_TYPE_OVERLAY);
+	if (ret) {
+		drm_err(drm_dev, "Failed to create overlay layer\n");
+		return ret;
+	}
+
+	ret = ma35_layer_create(priv, NULL, 2, DRM_PLANE_TYPE_CURSOR);
+	if (ret) {
+		drm_err(drm_dev, "Failed to create cursor layer\n");
+		return ret;
+	}
+
+	return 0;
+}
diff --git a/drivers/gpu/drm/nuvoton/ma35_plane.h b/drivers/gpu/drm/nuvoton/ma35_plane.h
new file mode 100644
index 000000000000..b975590e03a5
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/ma35_plane.h
@@ -0,0 +1,226 @@
+/* SPDX-License-Identifier: GPL-2.0+ */
+/*
+ * Nuvoton DRM driver
+ *
+ * Copyright (C) 2026 Nuvoton Technology Corp.
+ *
+ * Author: Joey Lu <a0987203069@...il.com>
+ */
+
+#ifndef _MA35_LAYER_H_
+#define _MA35_LAYER_H_
+
+#include <linux/of.h>
+#include <linux/types.h>
+#include <drm/drm_plane.h>
+#include <drm/drm_property.h>
+
+#define MA35_MAX_PLANES	3
+
+struct ma35_drm;
+
+struct ma35_plane_property {
+	u32 fb_addr[MA35_MAX_PLANES];
+	u32 fb_stride[MA35_MAX_PLANES];
+	bool alpha;
+	bool swizzle;
+	bool colorkey;
+	bool background;
+	bool foreground;
+};
+
+struct ma35_layer_config {
+	u32 blend_mode;
+	u16 alpha_mode[2];
+	u32 src_color;
+	u32 dst_color;
+	u32 swizzle;
+	u32 colorkeylo;
+	u32 colorkeyup;
+	u32 background;
+	u32 foreground;
+	int32_t hotspot_x;
+	int32_t hotspot_y;
+};
+
+struct ma35_layer {
+	struct drm_plane drm_plane;
+	struct list_head list;
+	struct device_node *of_node;
+	struct ma35_layer_config config;
+	phys_addr_t fb_base[MA35_MAX_PLANES];
+
+	struct drm_property *blend_mode_prop;
+	struct drm_property *src_alpha_mode_prop;
+	struct drm_property *dst_alpha_mode_prop;
+	struct drm_property *src_color_prop;
+	struct drm_property *dst_color_prop;
+	struct drm_property *swizzle_prop;
+	struct drm_property *colorkeylo_prop;
+	struct drm_property *colorkeyup_prop;
+	struct drm_property *background_prop;
+	struct drm_property *foreground_prop;
+};
+
+enum ma35_format_enum {
+	MA35_FORMAT_X4R4G4B4,	// DRM_FORMAT_XRGB4444
+	MA35_FORMAT_A4R4G4B4,	// DRM_FORMAT_ARGB4444
+	MA35_FORMAT_X1R5G5B5,	// DRM_FORMAT_XRGB1555
+	MA35_FORMAT_A1R5G5B5,	// DRM_FORMAT_ARGB1555
+	MA35_FORMAT_R5G6B5,		// DRM_FORMAT_RGB565
+	MA35_FORMAT_X8R8G8B8,	// DRM_FORMAT_XRGB8888
+	MA35_FORMAT_A8R8G8B8,	// DRM_FORMAT_ARGB8888
+	MA35_FORMAT_YUY2,		// YUV422, DRM_FORMAT_YUYV
+	MA35_FORMAT_UYVY,		// YUV422, DRM_FORMAT_UYVY
+	MA35_FORMAT_INDEX8,
+	MA35_FORMAT_MONOCHROME,
+	MA35_FORMAT_YV12,		// YUV420, DRM_FORMAT_YVU420
+	MA35_FORMAT_A8,
+	MA35_FORMAT_NV12,		// YUV420, DRM_FORMAT_NV12
+	MA35_FORMAT_NV16,		// YUV422, DRM_FORMAT_NV16
+	MA35_FORMAT_RG16,
+	MA35_FORMAT_R8,
+	MA35_FORMAT_NV12_10BIT,
+	MA35_FORMAT_A2R10G10B10, // DRM_FORMAT_ARGB2101010
+	MA35_FORMAT_NV16_10BIT,
+	MA35_FORMAT_INDEX1,
+	MA35_FORMAT_INDEX2,
+	MA35_FORMAT_INDEX4,
+	MA35_FORMAT_P010,		// YUV420, DRM_FORMAT_P010
+	MA35_FORMAT_NV12_10BIT_L1,
+	MA35_FORMAT_NV16_10BIT_L1,
+};
+
+/* output = src * a + dst * b */
+enum ma35_blend_mode_enum { // (a, b)
+	MA35_ALPHA_CLEAR,	// (0, 0)
+	MA35_ALPHA_SRC,		// (1, 0)
+	MA35_ALPHA_DST,		// (0, 1)
+	MA35_ALPHA_SRC_OVER,	// (1, 1-alpha_s)
+	MA35_ALPHA_DST_OVER,	// (1-alpha_d, 1)
+	MA35_ALPHA_SRC_IN,	// (alpha_d, 0)
+	MA35_ALPHA_DST_IN,	// (0, alpha_s)
+	MA35_ALPHA_SRC_OUT,	// (1-alpha_d, 0)
+	MA35_ALPHA_DST_OUT,	// (0, 1-alpha_s)
+	MA35_ALPHA_SRC_ATOP,	// (alpha_d, 1-alpha_s)
+	MA35_ALPHA_DST_ATOP,	// (1-alpha_d, alpha_s)
+	MA35_ALPHA_XOR,		// (1-alpha_d, 1-alpha_s)
+};
+
+#define MA35_ALPHA_BLEND_DISABLE		BIT(1)
+
+#define MA35_SRC_ALPHA_MODE_INVERSED	BIT(0)
+#define MA35_DST_ALPHA_MODE_INVERSED	BIT(9)
+
+enum ma35_alpha_mode_enum {
+	MA35_ALPHA_MODE_NONE, // pass-through
+	MA35_ALPHA_MODE_GLOBAL, // substitute by global color
+};
+#define MA35_SRC_ALPHA_MODE		GENMASK(4, 3)
+#define MA35_DST_ALPHA_MODE		GENMASK(11, 10)
+
+enum ma35_alpha_blend_enum {
+	MA35_ALPHA_BLEND_ZERO,
+	MA35_ALPHA_BLEND_ONE,
+	MA35_ALPHA_BLEND_NORMAL,
+	MA35_ALPHA_BLEND_INVERSED,
+	MA35_ALPHA_BLEND_COLOR,
+	MA35_ALPHA_BLEND_COLOR_INVERSED,
+	MA35_ALPHA_BLEND_SATURATED_ALPHA,
+	MA35_ALPHA_BLEND_SATURATED_DEST_ALPHA,
+};
+#define MA35_SRC_BLENDING_MODE		GENMASK(7, 5)
+#define MA35_DST_BLENDING_MODE		GENMASK(14, 12)
+
+#define MA35_SRC_ALPHA_FACTOR_EN		BIT(8)
+#define MA35_DST_ALPHA_FACTOR_EN		BIT(15)
+
+/* configs for blend modes */
+#define MA35_BLEND_MODE_CLEAR	0
+#define MA35_BLEND_MODE_SRC	\
+	FIELD_PREP(MA35_SRC_BLENDING_MODE, MA35_ALPHA_BLEND_ONE)
+#define MA35_BLEND_MODE_DST \
+	FIELD_PREP(MA35_DST_BLENDING_MODE, MA35_ALPHA_BLEND_ONE)
+#define MA35_BLEND_MODE_SRC_OVER \
+	(FIELD_PREP(MA35_SRC_BLENDING_MODE, MA35_ALPHA_BLEND_ONE) | \
+	FIELD_PREP(MA35_DST_BLENDING_MODE, MA35_ALPHA_BLEND_INVERSED))
+#define MA35_BLEND_MODE_DST_OVER \
+	(FIELD_PREP(MA35_SRC_BLENDING_MODE, MA35_ALPHA_BLEND_INVERSED) | \
+	FIELD_PREP(MA35_DST_BLENDING_MODE, MA35_ALPHA_BLEND_ONE))
+#define MA35_BLEND_MODE_SRC_IN \
+	FIELD_PREP(MA35_SRC_BLENDING_MODE, MA35_ALPHA_BLEND_NORMAL)
+#define MA35_BLEND_MODE_DST_IN \
+	FIELD_PREP(MA35_DST_BLENDING_MODE, MA35_ALPHA_BLEND_NORMAL)
+#define MA35_BLEND_MODE_SRC_OUT \
+	FIELD_PREP(MA35_SRC_BLENDING_MODE, MA35_ALPHA_BLEND_INVERSED)
+#define MA35_BLEND_MODE_DST_OUT \
+	FIELD_PREP(MA35_DST_BLENDING_MODE, MA35_ALPHA_BLEND_INVERSED)
+#define MA35_BLEND_MODE_SRC_ATOP \
+	(FIELD_PREP(MA35_SRC_BLENDING_MODE, MA35_ALPHA_BLEND_NORMAL) | \
+	FIELD_PREP(MA35_DST_BLENDING_MODE, MA35_ALPHA_BLEND_INVERSED))
+#define MA35_BLEND_MODE_DST_ATOP \
+	(FIELD_PREP(MA35_SRC_BLENDING_MODE, MA35_ALPHA_BLEND_INVERSED) | \
+	FIELD_PREP(MA35_DST_BLENDING_MODE, MA35_ALPHA_BLEND_NORMAL))
+#define MA35_BLEND_MODE_XOR \
+	(FIELD_PREP(MA35_SRC_BLENDING_MODE, MA35_ALPHA_BLEND_INVERSED) | \
+	FIELD_PREP(MA35_DST_BLENDING_MODE, MA35_ALPHA_BLEND_INVERSED))
+
+/* colorkey */
+#define MA35_COLORKEY_ENABLE	2
+#define MA35_COLORKEY_DISABLE	0
+
+#define MA35_CURSOR_SIZE		32
+#define MA35_CURSOR_DEPTH		24
+
+enum ma35_cursor_formats_enum {
+	MA35_CURSOR_FORMAT_DISABLE,
+	MA35_CURSOR_FORMAT_MASKED,
+	MA35_CURSOR_FORMAT_A8R8G8B8,
+};
+
+#define MA35_CURSOR_FORMAT_MASK		GENMASK(1, 0)
+
+#define MA35_CURSOR_HOTSPOT_X_MASK	GENMASK(20, 16)
+#define MA35_CURSOR_HOTSPOT_Y_MASK	GENMASK(12, 8)
+#define MA35_CURSOR_FORMAT_MASK		GENMASK(1, 0)
+#define MA35_CURSOR_OWNER_MASK		BIT(4)
+#define MA35_CURSOR_X_MASK		GENMASK(14, 0)
+#define MA35_CURSOR_Y_MASK		GENMASK(30, 16)
+
+#define MA35_PRIMARY_ENABLE		BIT(0)
+#define MA35_PRIMARY_GAMMA		BIT(2)
+#define MA35_PRIMARY_RESET		BIT(4)
+#define MA35_PRIMARY_CLEAR		BIT(8)
+#define MA35_PRIMARY_TRANSPARENCY_MASK	GENMASK(10, 9)
+#define MA35_PRIMARY_SWIZZLE_MASK		GENMASK(25, 23)
+#define MA35_PRIMARY_FORMAT_MASK		GENMASK(31, 26)
+
+#define MA35_OVERLAY_ENABLE		BIT(24)
+#define MA35_OVERLAY_CLEAR		BIT(25)
+#define MA35_OVERLAY_FORMAT_MASK	GENMASK(21, 16)
+#define MA35_OVERLAY_SWIZZLE_MASK	GENMASK(15, 13)
+#define MA35_OVERLAY_TRANSPARENCY_MASK	GENMASK(1, 0)
+
+#define MA35_SWIZZLE_ARGB_MASK		GENMASK(1, 0)
+#define MA35_SWIZZLE_UV_MASK		BIT(2)
+
+enum ma35_swizzles_enum {
+	MA35_SWIZZLE_ARGB,
+	MA35_SWIZZLE_RGBA,
+	MA35_SWIZZLE_ABGR,
+	MA35_SWIZZLE_BGRA,
+	MA35_SWIZZLE_UV,
+};
+
+#define MA35_LAYER_FB_HEIGHT	GENMASK(29, 15)
+#define MA35_LAYER_FB_WIDTH		GENMASK(14, 0)
+
+#define MA35_OVERLAY_POSITION_Y_MASK	MA35_LAYER_FB_HEIGHT
+#define MA35_OVERLAY_POSITION_X_MASK	MA35_LAYER_FB_WIDTH
+
+struct ma35_layer *ma35_layer_get_from_type(struct ma35_drm *priv,
+							enum drm_plane_type type);
+void ma35_overlay_attach_crtc(struct ma35_drm *priv);
+int ma35_plane_init(struct ma35_drm *priv);
+
+#endif
diff --git a/drivers/gpu/drm/nuvoton/ma35_regs.h b/drivers/gpu/drm/nuvoton/ma35_regs.h
new file mode 100644
index 000000000000..0f4a7a13e7d8
--- /dev/null
+++ b/drivers/gpu/drm/nuvoton/ma35_regs.h
@@ -0,0 +1,88 @@
+/* SPDX-License-Identifier: GPL-2.0+ */
+/*
+ * Nuvoton DRM driver
+ *
+ * Copyright (C) 2026 Nuvoton Technology Corp.
+ *
+ * Author: Joey Lu <a0987203069@...il.com>
+ */
+
+#ifndef _MA35_REGS_H_
+#define _MA35_REGS_H_
+
+#define MA35_FRAMEBUFFER_CONFIG                   0x1518
+#define MA35_FRAMEBUFFER_ADDRESS                  0x1400
+#define MA35_FRAMEBUFFER_STRIDE                   0x1408
+#define MA35_HDISPLAY                             0x1430
+#define MA35_HSYNC                                0x1438
+#define MA35_VDISPLAY                             0x1440
+#define MA35_VSYNC                                0x1448
+#define MA35_PANEL_CONFIG                         0x1418
+#define MA35_DPI_CONFIG                           0x14B8
+#define MA35_CURSOR_ADDRESS                       0x146C
+#define MA35_CURSOR_CONFIG                        0x1468
+#define MA35_CURSOR_LOCATION                      0x1470
+#define MA35_CURSOR_BACKGROUND                    0x1474
+#define MA35_CURSOR_FOREGROUND                    0x1478
+#define MA35_FRAMEBUFFER_UPLANAR_ADDRESS          0x1530
+#define MA35_FRAMEBUFFER_VPLANAR_ADDRESS          0x1538
+#define MA35_FRAMEBUFFER_USTRIDE                  0x1800
+#define MA35_FRAMEBUFFER_VSTRIDE                  0x1808
+#define MA35_INDEXCOLOR_TABLEINDEX                0x1818
+#define MA35_INDEXCOLOR_TABLEDATA                 0x1820
+#define MA35_FRAMEBUFFER_SIZE                     0x1810
+#define MA35_FRAMEBUFFER_SCALEFACTORX             0x1828
+#define MA35_FRAMEBUFFER_SCALEFACTORY             0x1830
+#define MA35_FRAMEBUFFER_SCALEFCONFIG             0x1520
+#define MA35_HORIFILTER_KERNELINDEX               0x1838
+#define MA35_HORIFILTER_KERNEL                    0x1A00
+#define MA35_VERTIFILTER_KERNELINDEX              0x1A08
+#define MA35_VERTIFILTER_KERNEL                   0x1A10
+#define MA35_FRAMEBUFFER_INITIALOFFSET            0x1A20
+#define MA35_FRAMEBUFFER_COLORKEY                 0x1508
+#define MA35_FRAMEBUFFER_COLORHIGHKEY             0x1510
+#define MA35_FRAMEBUFFER_BGCOLOR                  0x1528
+#define MA35_FRAMEBUFFER_CLEARVALUE               0x1A18
+#define MA35_DISPLAY_INTRENABLE                   0x1480
+#define MA35_INT_STATE                            0x147C
+#define MA35_PANEL_DEST_ADDRESS                   0x14F0
+#define MA35_MEM_DEST_ADDRESS                     0x14E8
+#define MA35_DEST_CONFIG                          0x14F8
+#define MA35_DEST_STRIDE                          0x1500
+#define MA35_DBI_CONFIG                           0x1488
+#define MA35_AQHICLOCKCONTROL                     0x0000
+#define MA35_OVERLAY_CONFIG                       0x1540
+#define MA35_OVERLAY_STRIDE                       0x1600
+#define MA35_OVERLAY_USTRIDE                      0x18C0
+#define MA35_OVERLAY_VSTRIDE                      0x1900
+#define MA35_OVERLAY_TL                           0x1640
+#define MA35_OVERLAY_BR                           0x1680
+#define MA35_OVERLAY_ALPHA_BLEND_CONFIG           0x1580
+#define MA35_OVERLAY_SRC_GLOBAL_COLOR             0x16C0
+#define MA35_OVERLAY_DST_GLOBAL_COLOR             0x1700
+#define MA35_OVERLAY_CLEAR_VALUE                  0x1940
+#define MA35_OVERLAY_SIZE                         0x17C0
+#define MA35_OVERLAY_COLOR_KEY                    0x1740
+#define MA35_OVERLAY_COLOR_KEY_HIGH               0x1780
+#define MA35_OVERLAY_ADDRESS                      0x15C0
+#define MA35_OVERLAY_UPLANAR_ADDRESS              0x1840
+#define MA35_OVERLAY_VPLANAR_ADDRESS              0x1880
+#define MA35_OVERLAY_SCALE_CONFIG                 0x1C00
+#define MA35_OVERLAY_SCALE_FACTOR_X               0x1A40
+#define MA35_OVERLAY_SCALE_FACTOR_Y               0x1A80
+#define MA35_OVERLAY_HORI_FILTER_KERNEL_INDEX     0x1AC0
+#define MA35_OVERLAY_HORI_FILTER_KERNEL           0x1B00
+#define MA35_OVERLAY_VERTI_FILTER_KERNEL_INDEX    0x1B40
+#define MA35_OVERLAY_VERTI_FILTER_KERNEL          0x1B80
+#define MA35_OVERLAY_INITIAL_OFFSET               0x1BC0
+#define MA35_GAMMA_EX_INDEX                       0x1CF0
+#define MA35_GAMMA_EX_DATA                        0x1CF8
+#define MA35_GAMMA_EX_ONE_DATA                    0x1D80
+#define MA35_GAMMA_INDEX                          0x1458
+#define MA35_GAMMA_DATA                           0x1460
+#define MA35_DISPLAY_DITHER_TABLE_LOW             0x1420
+#define MA35_DISPLAY_DITHER_TABLE_HIGH            0x1428
+#define MA35_DISPLAY_DITHER_CONFIG                0x1410
+#define MA35_DISPLAY_CURRENT_LOCATION             0x1450
+
+#endif
-- 
2.43.0


Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ