drm/nvd0/disp: skeletal handling of modeset interrupts
authorBen Skeggs <bskeggs@redhat.com>
Tue, 5 Jul 2011 04:16:05 +0000 (14:16 +1000)
committerBen Skeggs <bskeggs@redhat.com>
Tue, 20 Sep 2011 06:06:24 +0000 (16:06 +1000)
Signed-off-by: Ben Skeggs <bskeggs@redhat.com>
drivers/gpu/drm/nouveau/nvd0_display.c

index 3a2a4bb1276de153abb3fbdab4a96b4bb260a846..65e48f953d4c74fa77992e8be911330149e6f74b 100644 (file)
@@ -268,6 +268,42 @@ nvd0_sor_create(struct drm_connector *connector, struct dcb_entry *dcbe)
 /******************************************************************************
  * IRQ
  *****************************************************************************/
+static void
+nvd0_display_unk1_handler(struct drm_device *dev)
+{
+       u32 unk0 = nv_rd32(dev, 0x6101d0);
+
+       NV_INFO(dev, "PDISP: unk1 0x%08x\n", unk0);
+
+       nv_wr32(dev, 0x6101d4, 0x00000000);
+       nv_wr32(dev, 0x6109d4, 0x00000000);
+       nv_wr32(dev, 0x6101d0, 0x80000000);
+}
+
+static void
+nvd0_display_unk2_handler(struct drm_device *dev)
+{
+       u32 unk0 = nv_rd32(dev, 0x6101d0);
+
+       NV_INFO(dev, "PDISP: unk2 0x%08x\n", unk0);
+
+       nv_wr32(dev, 0x6101d4, 0x00000000);
+       nv_wr32(dev, 0x6109d4, 0x00000000);
+       nv_wr32(dev, 0x6101d0, 0x80000000);
+}
+
+static void
+nvd0_display_unk4_handler(struct drm_device *dev)
+{
+       u32 unk0 = nv_rd32(dev, 0x6101d0);
+
+       NV_INFO(dev, "PDISP: unk4 0x%08x\n", unk0);
+
+       nv_wr32(dev, 0x6101d4, 0x00000000);
+       nv_wr32(dev, 0x6109d4, 0x00000000);
+       nv_wr32(dev, 0x6101d0, 0x80000000);
+}
+
 static void
 nvd0_display_intr(struct drm_device *dev)
 {
@@ -291,6 +327,29 @@ nvd0_display_intr(struct drm_device *dev)
                intr &= ~0x00000002;
        }
 
+       if (intr & 0x00100000) {
+               u32 stat = nv_rd32(dev, 0x6100ac);
+
+               if (stat & 0x00000007) {
+                       nv_wr32(dev, 0x6100ac, (stat & 0x00000007));
+
+                       if (stat & 0x00000001)
+                               nvd0_display_unk1_handler(dev);
+                       if (stat & 0x00000002)
+                               nvd0_display_unk2_handler(dev);
+                       if (stat & 0x00000004)
+                               nvd0_display_unk4_handler(dev);
+                       stat &= ~0x00000007;
+               }
+
+               if (stat) {
+                       NV_INFO(dev, "PDISP: unknown intr24 0x%08x\n", stat);
+                       nv_wr32(dev, 0x6100ac, stat);
+               }
+
+               intr &= ~0x00100000;
+       }
+
        if (intr & 0x01000000) {
                u32 stat = nv_rd32(dev, 0x6100bc);
                nv_wr32(dev, 0x6100bc, stat);
@@ -354,6 +413,7 @@ nvd0_display_init(struct drm_device *dev)
        }
 
        nv_wr32(dev, 0x610010, (disp->mem->vinst >> 8) | 9);
+       nv_mask(dev, 0x6100b0, 0x00000307, 0x00000307);
 
        /* init master */
        nv_wr32(dev, 0x610494, (disp->evo[0].handle >> 8) | 3);