Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch/arm/sunxi Add support for vblank irq and RGB overla...



details:   https://anonhg.NetBSD.org/src/rev/5f0e374415c3
branches:  trunk
changeset: 996558:5f0e374415c3
user:      jmcneill <jmcneill%NetBSD.org@localhost>
date:      Mon Feb 04 12:10:13 2019 +0000

description:
Add support for vblank irq and RGB overlay planes.

diffstat:

 sys/arch/arm/sunxi/sunxi_drm.c   |   73 ++++++++++++-
 sys/arch/arm/sunxi/sunxi_drm.h   |   13 ++-
 sys/arch/arm/sunxi/sunxi_lcdc.c  |   98 ++++++++++++++++++-
 sys/arch/arm/sunxi/sunxi_mixer.c |  206 ++++++++++++++++++++++++++++++++++++++-
 4 files changed, 376 insertions(+), 14 deletions(-)

diffs (truncated from 644 to 300 lines):

diff -r b968d4d836ad -r 5f0e374415c3 sys/arch/arm/sunxi/sunxi_drm.c
--- a/sys/arch/arm/sunxi/sunxi_drm.c    Mon Feb 04 12:08:03 2019 +0000
+++ b/sys/arch/arm/sunxi/sunxi_drm.c    Mon Feb 04 12:10:13 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: sunxi_drm.c,v 1.5 2019/02/03 15:43:57 jmcneill Exp $ */
+/* $NetBSD: sunxi_drm.c,v 1.6 2019/02/04 12:10:13 jmcneill Exp $ */
 
 /*-
  * Copyright (c) 2019 Jared D. McNeill <jmcneill%invisible.ca@localhost>
@@ -27,7 +27,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: sunxi_drm.c,v 1.5 2019/02/03 15:43:57 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: sunxi_drm.c,v 1.6 2019/02/04 12:10:13 jmcneill Exp $");
 
 #include <sys/param.h>
 #include <sys/bus.h>
@@ -71,6 +71,10 @@
 
 static int     sunxi_drm_set_busid(struct drm_device *, struct drm_master *);
 
+static uint32_t        sunxi_drm_get_vblank_counter(struct drm_device *, unsigned int);
+static int     sunxi_drm_enable_vblank(struct drm_device *, unsigned int);
+static void    sunxi_drm_disable_vblank(struct drm_device *, unsigned int);
+
 static int     sunxi_drm_load(struct drm_device *, unsigned long);
 static int     sunxi_drm_unload(struct drm_device *);
 
@@ -88,11 +92,9 @@
        .dumb_map_offset = drm_gem_cma_dumb_map_offset,
        .dumb_destroy = drm_gem_dumb_destroy,
 
-#if notyet
        .get_vblank_counter = sunxi_drm_get_vblank_counter,
        .enable_vblank = sunxi_drm_enable_vblank,
        .disable_vblank = sunxi_drm_disable_vblank,
-#endif
 
        .name = DRIVER_NAME,
        .desc = DRIVER_DESC,
@@ -233,12 +235,23 @@
        fb = kmem_zalloc(sizeof(*fb), KM_SLEEP);
        fb->obj = to_drm_gem_cma_obj(gem_obj);
        fb->base.pitches[0] = cmd->pitches[0];
+       fb->base.pitches[1] = cmd->pitches[1];
+       fb->base.pitches[2] = cmd->pitches[2];
        fb->base.offsets[0] = cmd->offsets[0];
+       fb->base.offsets[1] = cmd->offsets[2];
+       fb->base.offsets[2] = cmd->offsets[1];
        fb->base.width = cmd->width;
        fb->base.height = cmd->height;
        fb->base.pixel_format = cmd->pixel_format;
-       drm_fb_get_bpp_depth(cmd->pixel_format, &fb->base.depth,
-           &fb->base.bits_per_pixel);
+       fb->base.bits_per_pixel = drm_format_plane_cpp(fb->base.pixel_format, 0) * 8;
+
+       switch (fb->base.pixel_format) {
+       case DRM_FORMAT_XRGB8888:
+               fb->base.depth = 32;
+               break;
+       default:
+               break;
+       }
 
        error = drm_framebuffer_init(ddev, &fb->base, &sunxi_drm_framebuffer_funcs);
        if (error != 0)
@@ -372,6 +385,10 @@
 
        drm_fb_helper_initial_config(&fbdev->helper, 32);
 
+       /* XXX */
+       ddev->irq_enabled = true;
+       drm_vblank_init(ddev, num_crtc);
+
        return 0;
 
 drmerr:
@@ -381,6 +398,50 @@
        return error;
 }
 
+static uint32_t
+sunxi_drm_get_vblank_counter(struct drm_device *ddev, unsigned int crtc)
+{
+       struct sunxi_drm_softc * const sc = sunxi_drm_private(ddev);
+
+       if (crtc >= __arraycount(sc->sc_vbl))
+               return 0;
+
+       if (sc->sc_vbl[crtc].get_vblank_counter == NULL)
+               return 0;
+
+       return sc->sc_vbl[crtc].get_vblank_counter(sc->sc_vbl[crtc].priv);
+}
+
+static int
+sunxi_drm_enable_vblank(struct drm_device *ddev, unsigned int crtc)
+{
+       struct sunxi_drm_softc * const sc = sunxi_drm_private(ddev);
+
+       if (crtc >= __arraycount(sc->sc_vbl))
+               return 0;
+
+       if (sc->sc_vbl[crtc].enable_vblank == NULL)
+               return 0;
+
+       sc->sc_vbl[crtc].enable_vblank(sc->sc_vbl[crtc].priv);
+
+       return 0;
+}
+
+static void
+sunxi_drm_disable_vblank(struct drm_device *ddev, unsigned int crtc)
+{
+       struct sunxi_drm_softc * const sc = sunxi_drm_private(ddev);
+
+       if (crtc >= __arraycount(sc->sc_vbl))
+               return;
+
+       if (sc->sc_vbl[crtc].disable_vblank == NULL)
+               return;
+
+       sc->sc_vbl[crtc].disable_vblank(sc->sc_vbl[crtc].priv);
+}
+
 static int
 sunxi_drm_unload(struct drm_device *ddev)
 {
diff -r b968d4d836ad -r 5f0e374415c3 sys/arch/arm/sunxi/sunxi_drm.h
--- a/sys/arch/arm/sunxi/sunxi_drm.h    Mon Feb 04 12:08:03 2019 +0000
+++ b/sys/arch/arm/sunxi/sunxi_drm.h    Mon Feb 04 12:10:13 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: sunxi_drm.h,v 1.1 2019/01/30 01:24:00 jmcneill Exp $ */
+/* $NetBSD: sunxi_drm.h,v 1.2 2019/02/04 12:10:13 jmcneill Exp $ */
 
 /*-
  * Copyright (c) 2019 Jared D. McNeill <jmcneill%invisible.ca@localhost>
@@ -44,6 +44,15 @@
 
 struct sunxi_framebuffer;
 
+#define        SUNXI_DRM_MAX_CRTC      2
+
+struct sunxi_drm_vblank {
+       void                    *priv;
+       void                    (*enable_vblank)(void *);
+       void                    (*disable_vblank)(void *);
+       uint32_t                (*get_vblank_counter)(void *);
+};
+
 struct sunxi_drm_softc {
        device_t                sc_dev;
        struct drm_device       *sc_ddev;
@@ -52,6 +61,8 @@
        bus_dma_tag_t           sc_dmat;
 
        int                     sc_phandle;
+
+       struct sunxi_drm_vblank sc_vbl[SUNXI_DRM_MAX_CRTC];
 };
 
 struct sunxi_drm_framebuffer {
diff -r b968d4d836ad -r 5f0e374415c3 sys/arch/arm/sunxi/sunxi_lcdc.c
--- a/sys/arch/arm/sunxi/sunxi_lcdc.c   Mon Feb 04 12:08:03 2019 +0000
+++ b/sys/arch/arm/sunxi/sunxi_lcdc.c   Mon Feb 04 12:10:13 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: sunxi_lcdc.c,v 1.3 2019/02/03 13:15:19 jmcneill Exp $ */
+/* $NetBSD: sunxi_lcdc.c,v 1.4 2019/02/04 12:10:13 jmcneill Exp $ */
 
 /*-
  * Copyright (c) 2019 Jared D. McNeill <jmcneill%invisible.ca@localhost>
@@ -27,7 +27,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: sunxi_lcdc.c,v 1.3 2019/02/03 13:15:19 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: sunxi_lcdc.c,v 1.4 2019/02/04 12:10:13 jmcneill Exp $");
 
 #include <sys/param.h>
 #include <sys/bus.h>
@@ -43,11 +43,17 @@
 #include <dev/fdt/fdtvar.h>
 #include <dev/fdt/fdt_port.h>
 
+#include <arm/sunxi/sunxi_drm.h>
+
 #define        TCON_GCTL_REG           0x000
 #define         TCON_GCTL_TCON_EN                      __BIT(31)
 #define         TCON_GCTL_GAMMA_EN                     __BIT(30)
 #define         TCON_GCTL_IO_MAP_SEL                   __BIT(0)
 #define        TCON_GINT0_REG          0x004
+#define         TCON_GINT0_TCON0_VB_INT_EN             __BIT(31)
+#define         TCON_GINT0_TCON1_VB_INT_EN             __BIT(30)
+#define         TCON_GINT0_TCON0_VB_INT_FLAG           __BIT(15)
+#define         TCON_GINT0_TCON1_VB_INT_FLAG           __BIT(14)
 #define        TCON_GINT1_REG          0x008
 #define         TCON_GINT1_TCON1_LINE_INT_NUM          __BITS(11,0)
 
@@ -129,6 +135,8 @@
        struct drm_connector    sc_connector;
 
        struct fdt_device_ports sc_ports;
+
+       uint32_t                sc_vbl_counter;
 };
 
 #define        to_sunxi_lcdc_encoder(x)        container_of(x, struct sunxi_lcdc_encoder, base)
@@ -336,6 +344,50 @@
        }
 }
 
+static uint32_t
+sunxi_lcdc_get_vblank_counter(void *priv)
+{
+       struct sunxi_lcdc_softc * const sc = priv;
+
+       return sc->sc_vbl_counter;
+}
+
+static void
+sunxi_lcdc_enable_vblank(void *priv)
+{
+       struct sunxi_lcdc_softc * const sc = priv;
+        const int crtc_index = ffs32(sc->sc_encoder.base.possible_crtcs) - 1;
+
+       if (crtc_index == 0)
+               TCON_WRITE(sc, TCON_GINT0_REG, TCON_GINT0_TCON0_VB_INT_EN);
+       else
+               TCON_WRITE(sc, TCON_GINT0_REG, TCON_GINT0_TCON1_VB_INT_EN);
+}
+
+static void
+sunxi_lcdc_disable_vblank(void *priv)
+{
+       struct sunxi_lcdc_softc * const sc = priv;
+
+       TCON_WRITE(sc, TCON_GINT0_REG, 0);
+}
+
+static void
+sunxi_lcdc_setup_vblank(struct sunxi_lcdc_softc *sc)
+{
+        const int crtc_index = ffs32(sc->sc_encoder.base.possible_crtcs) - 1;
+       struct drm_device *ddev = sc->sc_encoder.base.dev;
+       struct sunxi_drm_softc *drm_sc;
+
+       KASSERT(ddev != NULL);
+
+       drm_sc = device_private(ddev->dev);
+       drm_sc->sc_vbl[crtc_index].priv = sc;
+       drm_sc->sc_vbl[crtc_index].get_vblank_counter = sunxi_lcdc_get_vblank_counter;
+       drm_sc->sc_vbl[crtc_index].enable_vblank = sunxi_lcdc_enable_vblank;
+       drm_sc->sc_vbl[crtc_index].disable_vblank = sunxi_lcdc_disable_vblank;
+}
+
 static int
 sunxi_lcdc_ep_activate(device_t dev, struct fdt_endpoint *ep, bool activate)
 {
@@ -364,6 +416,8 @@
                    sunxi_lcdc_encoder_mode(out_ep));
                drm_encoder_helper_add(&sc->sc_encoder.base, &sunxi_lcdc_tcon0_helper_funcs);
 
+               sunxi_lcdc_setup_vblank(sc);
+
                return fdt_endpoint_activate(out_ep, activate);
        }
 
@@ -373,6 +427,8 @@
                    sunxi_lcdc_encoder_mode(out_ep));
                drm_encoder_helper_add(&sc->sc_encoder.base, &sunxi_lcdc_tcon1_helper_funcs);
 
+               sunxi_lcdc_setup_vblank(sc);
+
                return fdt_endpoint_activate(out_ep, activate);
        }
 
@@ -388,6 +444,28 @@
 }
 
 static int
+sunxi_lcdc_intr(void *priv)
+{
+       struct sunxi_lcdc_softc * const sc = priv;
+       uint32_t val;
+       int rv = 0;
+
+       const int crtc_index = ffs32(sc->sc_encoder.base.possible_crtcs) - 1;
+       const uint32_t status_mask = crtc_index == 0 ?
+           TCON_GINT0_TCON0_VB_INT_FLAG : TCON_GINT0_TCON1_VB_INT_FLAG;
+
+       val = TCON_READ(sc, TCON_GINT0_REG);
+       if ((val & status_mask) != 0) {
+               TCON_WRITE(sc, TCON_GINT0_REG, val & ~status_mask);
+               atomic_inc_32(&sc->sc_vbl_counter);
+               drm_handle_vblank(sc->sc_encoder.base.dev, crtc_index);
+               rv = 1;
+       }
+
+       return rv;



Home | Main Index | Thread Index | Old Index