add bcm accel
authorFelix Domke <tmbinc@elitedvb.net>
Wed, 3 Jun 2009 16:28:38 +0000 (18:28 +0200)
committerFelix Domke <tmbinc@elitedvb.net>
Wed, 3 Jun 2009 16:28:38 +0000 (18:28 +0200)
lib/gdi/Makefile.am
lib/gdi/accel.cpp
lib/gdi/accel.h
lib/gdi/bcm.cpp [new file with mode: 0644]

index e5430fab94d56b3b121b8302ec4c7c0255645fa2..1280556eff8d47bae2c5a782accd426e9a3ddb55 100644 (file)
@@ -6,7 +6,7 @@ noinst_LIBRARIES = libenigma_gdi.a
 libenigma_gdi_a_SOURCES = \
        region.cpp grc.cpp epng.cpp erect.cpp fb.cpp font.cpp font_arabic.cpp gfbdc.cpp  \
        glcddc.cpp gpixmap.cpp lcd.cpp gfont.cpp accel.cpp picload.cpp picexif.cpp \
-       compositing.cpp
+       compositing.cpp bcm.cpp
 
 if WITH_SDL
 libenigma_gdi_a_SOURCES += sdl.cpp
index f1ffe193502d6847a1ffb31b7df1db857abc2420..bf9a5a44d26f3039aa908f67f3c382f36e879dde 100644 (file)
@@ -9,7 +9,7 @@
 #include <lib/gdi/gpixmap.h>
 
 gAccel *gAccel::instance;
-// #define BCM_ACCEL
+#define BCM_ACCEL
 
 #ifdef ATI_ACCEL
 extern int ati_accel_init(void);
@@ -50,7 +50,7 @@ gAccel::gAccel()
        ati_accel_init();
 #endif
 #ifdef BCM_ACCEL       
-       bcm_accel_init();
+       m_bcm_accel_state = bcm_accel_init();
 #endif
 }
 
@@ -95,12 +95,15 @@ int gAccel::blit(gSurface *dst, const gSurface *src, const eRect &p, const eRect
        return 0;
 #endif
 #ifdef BCM_ACCEL
-       bcm_accel_blit(
-               src->data_phys, src->x, src->y, src->stride,
-               dst->data_phys, dst->x, dst->y, dst->stride, 
-               area.left(), area.top(), area.width(), area.height(),
-               p.x(), p.y(), p.width(), p.height());
-       return 0;
+       if (!m_bcm_accel_state)
+       {
+               bcm_accel_blit(
+                       src->data_phys, src->x, src->y, src->stride,
+                       dst->data_phys, dst->x, dst->y, dst->stride, 
+                       area.left(), area.top(), area.width(), area.height(),
+                       p.x(), p.y(), p.width(), p.height());
+               return 0;
+       }
 #endif
        return -1;
 }
index b08ba6610defb44c83f1188b350a9cde7fc8deda..bbb2e26acb3ca17308123a2d50aca1e51ca535eb 100644 (file)
@@ -24,6 +24,7 @@ private:
        int m_accel_phys_addr;
        int m_accel_size; // in blocks
        int *m_accel_allocation;
+       int m_bcm_accel_state;
        
        static gAccel *instance;
 };
diff --git a/lib/gdi/bcm.cpp b/lib/gdi/bcm.cpp
new file mode 100644 (file)
index 0000000..bd82091
--- /dev/null
@@ -0,0 +1,116 @@
+/*
+  Interface to the Dreambox dm800/dm8000 proprietary accel interface.
+*/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <linux/fb.h>
+#include <sys/mman.h>
+#include <sys/ioctl.h>
+
+#define FBIO_ACCEL  0x23
+
+static unsigned int displaylist[1024];
+static int ptr;
+
+#define P(x, y) do { displaylist[ptr++] = x; displaylist[ptr++] = y; } while (0)
+#define C(x) P(x, 0)
+
+static int fb_fd;
+static int exec_list(void);
+
+int bcm_accel_init(void)
+{
+       fb_fd = open("/dev/fb/0", O_RDWR);
+       if (fb_fd < 0)
+       {
+               perror("/dev/fb/0");
+               return 1;
+       }
+       if (exec_list())
+       {
+               fprintf(stderr, "BCM accel interface not available - %m\n");
+               close(fb_fd);
+               return 1;
+       }
+       return 0;
+}
+
+void bcm_accel_close(void)
+{
+       close(fb_fd);
+}
+
+static int exec_list(void)
+{
+       int ret;
+       struct
+       {
+               void *ptr;
+               int len;
+       } l;
+
+       l.ptr = displaylist;
+       l.len = ptr;
+       ret = ioctl(fb_fd, FBIO_ACCEL, &l);
+       ptr = 0;
+       return ret;
+}
+
+void bcm_accel_blit(
+               int src_addr, int src_width, int src_height, int src_stride,
+               int dst_addr, int dst_width, int dst_height, int dst_stride,
+               int src_x, int src_y, int width, int height,
+               int dst_x, int dst_y, int dwidth, int dheight)
+{
+       C(0x43); // reset source
+       C(0x53); // reset dest
+       C(0x5b);  // reset pattern
+       C(0x67); // reset blend
+       C(0x75); // reset output
+
+       P(0x0, src_addr); // set source addr
+       P(0x1, src_stride);  // set source pitch
+       P(0x2, src_width); // source width
+       P(0x3, src_height); // height
+       P(0x4, 0x7e48888); // format: ARGB 8888
+
+       C(0x5); // set source surface (based on last parameters)
+
+       P(0x2e, src_x); // define  rect
+       P(0x2f, src_y);
+       P(0x30, width);
+       P(0x31, height);
+
+       C(0x32); // set this rect as source rect
+
+       P(0x0, dst_addr); // prepare output surface
+       P(0x1, dst_stride);
+       P(0x2, dst_width);
+       P(0x3, dst_height);
+       P(0x4, 0x7e48888);
+       
+       C(0x69); // set output surface
+       
+       P(0x2e, dst_x); // prepare output rect
+       P(0x2f, dst_y);
+       P(0x30, dwidth);
+       P(0x31, dheight);
+
+       C(0x6e); // set this rect as output rect
+
+       C(0x77);  // do it
+
+       exec_list();
+}
+
+void bcm_accel_fill(
+               int dst_addr, int dst_width, int dst_height, int dst_stride,
+               int x, int y, int width, int height,
+               unsigned long color)
+{
+//     printf("unimplemented bcm_accel_fill\n");
+}
+