bochsvga: fallback to stdvga if dispi interface isn't present

Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
diff --git a/vgasrc/bochsvga.c b/vgasrc/bochsvga.c
index 2002118..b93213c 100644
--- a/vgasrc/bochsvga.c
+++ b/vgasrc/bochsvga.c
@@ -91,6 +91,8 @@
     { 0x18c, { MM_DIRECT, 2560, 1600, 32, 8, 16, SEG_GRAPH } },
 };
 
+static int dispi_found VAR16 = 0;
+
 static int is_bochsvga_mode(struct vgamode_s *vmode_g)
 {
     return (vmode_g >= &bochsvga_modes[0].info
@@ -100,9 +102,10 @@
 struct vgamode_s *bochsvga_find_mode(int mode)
 {
     struct bochsvga_mode *m = bochsvga_modes;
-    for (; m < &bochsvga_modes[ARRAY_SIZE(bochsvga_modes)]; m++)
-        if (GET_GLOBAL(m->mode) == mode)
-            return &m->info;
+    if (GET_GLOBAL(dispi_found))
+        for (; m < &bochsvga_modes[ARRAY_SIZE(bochsvga_modes)]; m++)
+            if (GET_GLOBAL(m->mode) == mode)
+                return &m->info;
     return stdvga_find_mode(mode);
 }
 
@@ -110,12 +113,14 @@
 bochsvga_list_modes(u16 seg, u16 *dest, u16 *last)
 {
     struct bochsvga_mode *m = bochsvga_modes;
-    for (; m < &bochsvga_modes[ARRAY_SIZE(bochsvga_modes)] && dest<last; m++) {
-        u16 mode = GET_GLOBAL(m->mode);
-        if (mode == 0xffff)
-            continue;
-        SET_FARVAR(seg, *dest, mode);
-        dest++;
+    if (GET_GLOBAL(dispi_found)) {
+        for (; m < &bochsvga_modes[ARRAY_SIZE(bochsvga_modes)] && dest<last; m++) {
+            u16 mode = GET_GLOBAL(m->mode);
+            if (mode == 0xffff)
+                continue;
+            SET_FARVAR(seg, *dest, mode);
+            dest++;
+        }
     }
     stdvga_list_modes(seg, dest, last);
 }
@@ -128,6 +133,8 @@
 int
 bochsvga_get_window(struct vgamode_s *vmode_g, int window)
 {
+    if (!GET_GLOBAL(dispi_found))
+        return stdvga_get_window(vmode_g, window);
     if (window != 0)
         return -1;
     return dispi_read(VBE_DISPI_INDEX_BANK);
@@ -136,6 +143,8 @@
 int
 bochsvga_set_window(struct vgamode_s *vmode_g, int window, int val)
 {
+    if (!GET_GLOBAL(dispi_found))
+        return stdvga_set_window(vmode_g, window, val);
     if (window != 0)
         return -1;
     dispi_write(VBE_DISPI_INDEX_BANK, val);
@@ -147,6 +156,8 @@
 int
 bochsvga_get_linelength(struct vgamode_s *vmode_g)
 {
+    if (!GET_GLOBAL(dispi_found))
+        return stdvga_get_linelength(vmode_g);
     return dispi_read(VBE_DISPI_INDEX_VIRT_WIDTH) * vga_bpp(vmode_g) / 8;
 }
 
@@ -154,14 +165,18 @@
 bochsvga_set_linelength(struct vgamode_s *vmode_g, int val)
 {
     stdvga_set_linelength(vmode_g, val);
-    int pixels = (val * 8) / vga_bpp(vmode_g);
-    dispi_write(VBE_DISPI_INDEX_VIRT_WIDTH, pixels);
+    if (GET_GLOBAL(dispi_found)) {
+        int pixels = (val * 8) / vga_bpp(vmode_g);
+        dispi_write(VBE_DISPI_INDEX_VIRT_WIDTH, pixels);
+    }
     return 0;
 }
 
 int
 bochsvga_get_displaystart(struct vgamode_s *vmode_g)
 {
+    if (!GET_GLOBAL(dispi_found))
+        return stdvga_get_displaystart(vmode_g);
     int bpp = vga_bpp(vmode_g);
     int linelength = dispi_read(VBE_DISPI_INDEX_VIRT_WIDTH) * bpp / 8;
     int x = dispi_read(VBE_DISPI_INDEX_X_OFFSET);
@@ -173,16 +188,20 @@
 bochsvga_set_displaystart(struct vgamode_s *vmode_g, int val)
 {
     stdvga_set_displaystart(vmode_g, val);
-    int bpp = vga_bpp(vmode_g);
-    int linelength = dispi_read(VBE_DISPI_INDEX_VIRT_WIDTH) * bpp / 8;
-    dispi_write(VBE_DISPI_INDEX_X_OFFSET, (val % linelength) * 8 / bpp);
-    dispi_write(VBE_DISPI_INDEX_Y_OFFSET, val / linelength);
+    if (GET_GLOBAL(dispi_found)) {
+        int bpp = vga_bpp(vmode_g);
+        int linelength = dispi_read(VBE_DISPI_INDEX_VIRT_WIDTH) * bpp / 8;
+        dispi_write(VBE_DISPI_INDEX_X_OFFSET, (val % linelength) * 8 / bpp);
+        dispi_write(VBE_DISPI_INDEX_Y_OFFSET, val / linelength);
+    }
     return 0;
 }
 
 int
 bochsvga_get_dacformat(struct vgamode_s *vmode_g)
 {
+    if (!GET_GLOBAL(dispi_found))
+        return stdvga_get_dacformat(vmode_g);
     u16 en = dispi_read(VBE_DISPI_INDEX_ENABLE);
     return (en & VBE_DISPI_8BIT_DAC) ? 8 : 6;
 }
@@ -190,6 +209,8 @@
 int
 bochsvga_set_dacformat(struct vgamode_s *vmode_g, int val)
 {
+    if (!GET_GLOBAL(dispi_found))
+        return stdvga_set_dacformat(vmode_g, val);
     u16 en = dispi_read(VBE_DISPI_INDEX_ENABLE);
     if (val == 6)
         en &= ~VBE_DISPI_8BIT_DAC;
@@ -207,7 +228,7 @@
     int size = stdvga_size_state(states);
     if (size < 0)
         return size;
-    if (states & 8)
+    if (GET_GLOBAL(dispi_found) && (states & 8))
         size += (VBE_DISPI_INDEX_Y_OFFSET-VBE_DISPI_INDEX_XRES+1)*sizeof(u16);
     return size;
 }
@@ -219,6 +240,8 @@
     if (ret < 0)
         return ret;
 
+    if (!GET_GLOBAL(dispi_found))
+        return 0;
     if (!(states & 8))
         return 0;
 
@@ -245,6 +268,8 @@
     if (ret < 0)
         return ret;
 
+    if (!GET_GLOBAL(dispi_found))
+        return 0;
     if (!(states & 8))
         return 0;
 
@@ -274,9 +299,12 @@
 int
 bochsvga_set_mode(struct vgamode_s *vmode_g, int flags)
 {
-    dispi_write(VBE_DISPI_INDEX_ENABLE, VBE_DISPI_DISABLED);
+    if (GET_GLOBAL(dispi_found))
+        dispi_write(VBE_DISPI_INDEX_ENABLE, VBE_DISPI_DISABLED);
     if (! is_bochsvga_mode(vmode_g))
         return stdvga_set_mode(vmode_g, flags);
+    if (!GET_GLOBAL(dispi_found))
+        return -1;
 
     u8 depth = GET_GLOBAL(vmode_g->depth);
     if (depth == 4)
@@ -339,11 +367,12 @@
     /* Sanity checks */
     dispi_write(VBE_DISPI_INDEX_ID, VBE_DISPI_ID0);
     if (dispi_read(VBE_DISPI_INDEX_ID) != VBE_DISPI_ID0) {
-        dprintf(1, "No VBE DISPI interface detected\n");
-        return -1;
+        dprintf(1, "No VBE DISPI interface detected, falling back to stdvga\n");
+        return 0;
     }
 
     dispi_write(VBE_DISPI_INDEX_ID, VBE_DISPI_ID5);
+    SET_VGA(dispi_found, 1);
 
     if (GET_GLOBAL(HaveRunInit))
         return 0;