[MPlayer-dev-eng] [PATCH] libcaca native support

Pigeon pigeon at pigeond.net
Fri Mar 26 00:30:24 CET 2004


Hi,

    I've done a patch for mplayer to use libcaca natively, i.e. not via
SDL, with very simple osd support.

    Built and tested with mplayer cvs, and libcaca head source.

Pigeon.

-------------- next part --------------
diff -ruN main/AUTHORS main-hacked/AUTHORS
--- main/AUTHORS	Fri Mar 26 08:46:21 2004
+++ main-hacked/AUTHORS	Fri Mar 26 09:59:29 2004
@@ -189,6 +189,9 @@
 Folke Ashberg <folke at ashberg.de>
     * native AAlib driver (-vo aa)
 
+Howell Tam (Pigeon) <pigeon at pigeond.net>
+    * native libcaca driver (-vo caca)
+
 Ivan Kalvachev <iive at yahoo.com>
     * interlaced MPEG2 support (libmpeg2)
     * libvo2 draft
diff -ruN main/Makefile main-hacked/Makefile
--- main/Makefile	Wed Dec 10 23:03:54 2003
+++ main-hacked/Makefile	Fri Mar 26 09:52:01 2004
@@ -32,7 +32,7 @@
 OBJS_MENCODER = $(SRCS_MENCODER:.c=.o)
 OBJS_MPLAYER = $(SRCS_MPLAYER:.c=.o)
 
-VO_LIBS = $(AA_LIB) $(X_LIB) $(SDL_LIB) $(GGI_LIB) $(MP1E_LIB) $(MLIB_LIB) $(SVGA_LIB) $(DIRECTFB_LIB) 
+VO_LIBS = $(AA_LIB) $(X_LIB) $(SDL_LIB) $(GGI_LIB) $(MP1E_LIB) $(MLIB_LIB) $(SVGA_LIB) $(DIRECTFB_LIB) $(CACA_LIB)
 AO_LIBS = $(ARTS_LIB) $(ESD_LIB) $(NAS_LIB) $(SGIAUDIO_LIB)
 CODEC_LIBS = $(AV_LIB) $(FAME_LIB) $(MAD_LIB) $(VORBIS_LIB) $(THEORA_LIB) $(FAAD_LIB) $(LIBLZO_LIB) $(DECORE_LIB) $(XVID_LIB) $(PNG_LIB) $(Z_LIB) $(JPEG_LIB) $(ALSA_LIB) $(XMMS_LIB) $(MATROSKA_LIB) 
 COMMON_LIBS = libmpcodecs/libmpcodecs.a mp3lib/libMP3.a liba52/liba52.a libmpeg2/libmpeg2.a $(W32_LIB) $(DS_LIB) libaf/libaf.a libmpdemux/libmpdemux.a input/libinput.a postproc/libswscale.a osdep/libosdep.a $(DVDREAD_LIB) $(CODEC_LIBS) $(FREETYPE_LIB) $(TERMCAP_LIB) $(CDPARANOIA_LIB) $(MPLAYER_NETWORK_LIB) $(WIN32_LIB) $(GIF_LIB) $(MACOSX_FRAMEWORKS) $(SMBSUPPORT_LIB) $(FRIBIDI_LIB) $(FLAC_LIB) $(FONTCONFIG_LIB)
diff -ruN main/configure main-hacked/configure
--- main/configure	Sun Mar 14 03:25:08 2004
+++ main-hacked/configure	Fri Mar 26 09:52:01 2004
@@ -221,6 +221,7 @@
   --enable-svga          build with SVGAlib support [autodetect]
   --enable-sdl           build with SDL render support [autodetect]
   --enable-aa            build with AAlib render support [autodetect]
+  --enable-caca          build with CACA render support [autodetect]
   --enable-ggi           build with GGI render support [autodetect]
   --enable-directx       build with DirectX support [autodetect]
   --enable-dxr2          build with DXR2 render support [autodetect]
@@ -1136,6 +1137,7 @@
 _gl=auto
 _ggi=auto
 _aa=auto
+_caca=auto
 _svga=auto
 _vesa=auto
 _fbdev=auto
@@ -1281,6 +1283,8 @@
   --disable-ggi)	_ggi=no		;;
   --enable-aa)		_aa=yes		;;
   --disable-aa)		_aa=no		;;
+  --enable-caca)	_caca=yes	;;
+  --disable-caca)	_caca=no	;;
   --enable-svga)	_svga=yes	;;
   --disable-svga)	_svga=no	;;
   --enable-vesa)	_vesa=yes	;;
@@ -3269,6 +3273,26 @@
 fi
 echores "$_aa"
 
+echocheck "CACA"
+if test "$_caca" = auto ; then
+  cat > $TMPC << EOF
+#include <caca.h>
+int main(void) { (void) caca_init(); return 0; }
+EOF
+  _caca=no
+  cc_check `caca-config --libs` && _caca=yes
+fi
+if test "$_caca" = yes ; then
+  _def_caca='#define HAVE_CACA 1'
+  _ld_caca=`caca-config --libs`
+  _vosrc="$_vosrc vo_caca.c"
+  _vomodules="caca $_vomodules"
+else
+  _def_caca='#undef HAVE_CACA'
+  _novomodules="caca $_novomodules"
+fi
+echores "$_caca"
+
 
 echocheck "SVGAlib"
 if test "$_svga" = auto ; then
@@ -5898,6 +5922,7 @@
 SDL_LIB = $_ld_sdl
 SVGA_LIB = $_ld_svga
 AA_LIB = $_ld_aa
+CACA_LIB = $_ld_caca
 
 # audio output
 ALSA_LIB = $_ld_alsa
@@ -6531,6 +6556,7 @@
 $_def_vesa
 $_def_xdpms
 $_def_aa
+$_def_caca
 $_def_tga
 
 /* used by GUI: */
diff -ruN main/libvo/video_out.c main-hacked/libvo/video_out.c
--- main/libvo/video_out.c	Sun Mar 14 03:48:52 2004
+++ main-hacked/libvo/video_out.c	Fri Mar 26 09:52:01 2004
@@ -83,6 +83,7 @@
 extern vo_functions_t video_out_png;
 extern vo_functions_t video_out_ggi;
 extern vo_functions_t video_out_aa;
+extern vo_functions_t video_out_caca;
 extern vo_functions_t video_out_mpegpes;
 extern vo_functions_t video_out_yuv4mpeg;
 #ifdef HAVE_DIRECTX
@@ -177,6 +178,9 @@
 #endif
 #ifdef HAVE_AA
 	&video_out_aa,
+#endif
+#ifdef HAVE_CACA
+	&video_out_caca,
 #endif
 #ifdef HAVE_DXR2
 	&video_out_dxr2,
diff -ruN main/libvo/vo_caca.c main-hacked/libvo/vo_caca.c
--- main/libvo/vo_caca.c	Thu Jan  1 10:00:00 1970
+++ main-hacked/libvo/vo_caca.c	Fri Mar 26 09:52:01 2004
@@ -0,0 +1,441 @@
+/*
+ * MPlayer
+ * 
+ * Video driver for libcaca
+ * 
+ * by Pigeon <pigeon at pigeond.net>
+ *
+ * Some functions/codes/ideas are from x11 and aalib vo
+ *
+ * TODO: support those draw_alpha stuff?
+ * 
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+
+#include <sys/stat.h>
+#include <unistd.h>
+
+#include <limits.h>
+#include <math.h>
+#include <stdarg.h>
+#include <time.h>
+#include <string.h>
+#include <errno.h>
+
+#include "config.h"
+#include "video_out.h"
+#include "video_out_internal.h"
+#include "aspect.h"
+#include "../postproc/swscale.h"
+#include "../libmpcodecs/vf_scale.h"
+#include "font_load.h"
+#include "sub.h"
+
+#include "osdep/keycodes.h"
+#include <caca.h>
+#include "m_option.h"
+#include "mp_msg.h"
+
+
+static vo_info_t info = {
+    "libcaca",
+    "caca",
+    "Pigeon <pigeon at pigeond.net>",
+    ""
+};
+
+LIBVO_EXTERN(caca)
+
+
+/* caca stuff */
+static struct caca_bitmap *cbitmap = NULL;
+
+
+/* image infos */
+static int image_format;
+static int image_width;
+static int image_height;
+
+static uint8_t *image = NULL;
+
+
+static int screen_w, screen_h;
+
+static struct SwsContext *sws = NULL;
+
+
+/* We want 24bpp always for now */
+static unsigned int bpp = 24;
+static unsigned int depth = 3;
+static unsigned int rmask = 0xff0000;
+static unsigned int gmask = 0x00ff00;
+static unsigned int bmask = 0x0000ff;
+static unsigned int amask = 0;
+
+
+#define MESSAGE_SIZE 512
+#define MESSAGE_DURATION 5
+
+static time_t stoposd = 0;
+static int showosdmessage = 0;
+static char osdmessagetext[MESSAGE_SIZE];
+static char posbar[MESSAGE_SIZE];
+
+static int osdx = 0, osdy = 0;
+static int posbary = 2;
+
+
+static void
+osdmessage(int duration, char *fmt, ...)
+{
+    /*
+     * for outputting a centered string at the bottom
+     * of our window for a while
+     */
+    va_list ar;
+    char m[MESSAGE_SIZE];
+
+    va_start(ar, fmt);
+    vsprintf(m, fmt, ar);
+    va_end(ar);
+    strcpy(osdmessagetext, m);
+
+    showosdmessage=1;
+    stoposd = time(NULL) + duration;
+    osdx = (screen_w - strlen(osdmessagetext)) / 2;
+    posbar[0]='\0';
+}
+
+static void
+osdpercent(int duration, int min, int max, int val, char *desc, char *unit)
+{
+    /*
+     * prints a bar for setting values
+     */
+    float step;
+    int where;
+    int i;
+
+
+    step = (float) screen_w / (float) (max - min);
+    where = (val - min) * step;
+    osdmessage(duration, "%s: %i%s", desc, val, unit);
+    posbar[0] = '|';
+    posbar[screen_w - 1] = '|';
+
+    for(i=0; i<screen_w; i++)
+    {
+        if(i == where)
+	{
+	    posbar[i] = '#';
+	}
+        else
+	{
+	    posbar[i] = '-';
+	}
+    }
+    if(where!=0)
+    {
+	posbar[0]='|';
+    }
+
+    if(where != (screen_w - 1))
+    {
+	posbar[screen_w - 1] = '|';
+    }
+
+    posbar[screen_w] = '\0';
+
+}
+
+
+static int
+resize()
+{
+    screen_w = caca_get_width();
+    screen_h = caca_get_height();
+
+    if(sws)
+    {
+	sws_freeContext(sws);
+    }
+
+    sws = sws_getContextFromCmdLine(image_width, image_height, image_format,
+	    image_width, image_height, IMGFMT_BGR24);
+
+    if(!sws)
+    {
+	mp_msg(MSGT_VO, MSGL_INFO, "vo_caca: couldn't do %x -> %x\n",
+		image_format, IMGFMT_BGR24);
+	return -1;
+    }
+
+    if(cbitmap)
+    {
+	caca_free_bitmap(cbitmap);
+    }
+
+    cbitmap = caca_create_bitmap(bpp, image_width, image_height,
+	    depth * image_width, rmask, gmask, bmask, amask);
+
+    if(image)
+    {
+	free(image);
+    }
+
+    image = (char *) malloc(image_width * image_height * depth * sizeof(char));
+
+    if(!image)
+    {
+	mp_msg(MSGT_VO, MSGL_FATAL, "vo_caca: malloc failed!\n");
+    }
+
+    return 0;
+}
+
+
+static uint32_t
+config(uint32_t width, uint32_t height, uint32_t d_width,
+	    uint32_t d_height, uint32_t fullscreen, char *title, 
+	    uint32_t format) {
+    /*
+     * main init
+     * called by mplayer
+     */
+
+    image_height = height;
+    image_width = width;
+    image_format = format;
+
+    mp_msg(MSGT_VO, MSGL_INFO,
+	    "\n"
+	    "MPlayer libcaca\n"
+	    "\n"
+	    "\tSome keys you might want to know about:\n"
+	    "\td - toggle libcaca dithering methods\n"
+	    "\ta - toggle libcaca antialiasing\n"
+	    "\n"
+	    "\tYou may also want to set libcaca environment variables such as:\n"
+	    "\tCACA_DRIVER: ncurses, slang, x11\n"
+	    "\tCACA_GEOMETRY: XxY, e.g. 128x50 (x11 only)\n"
+	    "\tCACA_FONT: e.g. fixed, nexus (x11 only)\n"
+	    "\n"
+	  );
+
+    return resize();
+}
+
+
+static uint32_t 
+draw_frame(uint8_t *src[]) {
+    mp_msg(MSGT_VO, MSGL_INFO, "vo_caca: draw_frame not implemented\n");
+    return -1;
+}
+
+static uint32_t 
+draw_slice(uint8_t *src[], int stride[], int w, int h, int x, int y) {
+
+    uint8_t *dst[3];
+    int dstStride[3];
+
+    dstStride[1] = dstStride[2] = 0;
+    dst[1] = dst[2] = NULL;
+
+    dstStride[0] = image_width * ((bpp + 7) / 8);
+    dst[0] = image;
+
+    if(sws)
+    {
+	sws_scale_ordered(sws, src, stride, y, h, dst, dstStride);
+    }
+
+    return 0;
+}
+
+
+static void 
+flip_page(void) {
+    caca_draw_bitmap(0, 0, screen_w, screen_h, cbitmap, image);
+
+    if(showosdmessage)
+    {
+	if(time(NULL) >= stoposd)
+	{
+	    showosdmessage = 0;
+
+	    if(posbar)
+	    {
+		posbar[0] = '\0';
+	    }
+	}
+	else
+	{
+	    caca_putstr(osdx, osdy, osdmessagetext);
+	    
+	    if(posbar)
+	    {
+		caca_putstr(0, posbary, posbar);
+	    }
+	}
+    }
+
+    caca_refresh();
+}
+
+static void 
+check_events(void) {
+
+    unsigned int cev;
+
+    if((cev = caca_get_event(CACA_EVENT_ANY)))
+    {
+	if(cev & CACA_EVENT_RESIZE)
+	{
+	    caca_refresh();
+	    resize();
+	}
+	else if(cev & CACA_EVENT_KEY_RELEASE)
+	{
+	    int key = (cev & 0x00ffffff);
+	    enum caca_feature cf;
+
+	    switch(key)
+	    {
+		case 'd':
+		case 'D':
+		    /* Toggle dithering method */
+		    cf = 1 + caca_get_feature(CACA_DITHERING);
+		    if(cf > CACA_DITHERING_MAX)
+		    {
+			cf = CACA_DITHERING_MIN;
+		    }
+		    caca_set_feature(cf);
+		    osdmessage(MESSAGE_DURATION,
+			    "Using %s", caca_get_feature_name(cf));
+		    break;
+
+		case 'a':
+		case 'A':
+		    /* Toggle antialiasing method */
+		    cf = 1 + caca_get_feature(CACA_ANTIALIASING);
+		    if(cf > CACA_ANTIALIASING_MAX)
+		    {
+			cf = CACA_ANTIALIASING_MIN;
+		    }
+		    caca_set_feature(cf);
+		    osdmessage(MESSAGE_DURATION,
+			    "Using %s", caca_get_feature_name(cf));
+		    break;
+
+		case CACA_KEY_UP:
+		    mplayer_put_key(KEY_UP);
+		    break;
+		case CACA_KEY_DOWN:
+		    mplayer_put_key(KEY_DOWN);
+		    break;
+		case CACA_KEY_LEFT:
+		    mplayer_put_key(KEY_LEFT);
+		    break;
+		case CACA_KEY_RIGHT:
+		    mplayer_put_key(KEY_RIGHT);
+		    break;
+		case CACA_KEY_ESCAPE:
+		    mplayer_put_key(KEY_ESC);
+		    break;
+		case CACA_KEY_PAGEUP:
+		    mplayer_put_key(KEY_PAGE_UP);
+		    break;
+		case CACA_KEY_PAGEDOWN:
+		    mplayer_put_key(KEY_PAGE_DOWN);
+		    break;
+		case CACA_KEY_RETURN:
+		    mplayer_put_key(KEY_ENTER);
+		    break;
+		case CACA_KEY_HOME:
+		    mplayer_put_key(KEY_HOME);
+		    break;
+		case CACA_KEY_END:
+		    mplayer_put_key(KEY_END);
+		    break;
+		default:
+		    if(key <= 255)
+		    {
+			mplayer_put_key(key);
+		    }
+		    break;
+	    }
+	}
+    }
+}
+
+static void 
+uninit(void) {
+    caca_free_bitmap(cbitmap);
+    cbitmap = NULL;
+    free(image);
+    image = NULL;
+    caca_end();
+}
+
+
+static void
+draw_osd(void){
+#ifdef USE_OSD
+    if(vo_osd_progbar_type != -1)
+    {
+	osdpercent(MESSAGE_DURATION, 0, 255,
+		vo_osd_progbar_value, __sub_osd_names[vo_osd_progbar_type], "");
+    }
+#endif
+}
+
+
+static uint32_t
+preinit(const char *arg)
+{
+    if(arg) 
+    {
+	mp_msg(MSGT_VO, MSGL_ERR, "vo_caca: Unknown subdevice: %s\n",arg);
+	return ENOSYS;
+    }
+
+    if (caca_init())
+    {
+	mp_msg(MSGT_VO, MSGL_ERR, "vo_caca: failed to initialize\n");
+    }
+
+    caca_set_window_title("mplayer");
+
+    /* Default libcaca features */
+    caca_set_feature(CACA_ANTIALIASING_PREFILTER);
+    caca_set_feature(CACA_DITHERING_RANDOM);
+
+    return 0;
+}
+
+static uint32_t 
+query_format(uint32_t format) {
+
+    if(format == IMGFMT_YV12)
+    {
+	return VFCAP_CSP_SUPPORTED | VFCAP_SWSCALE;
+    }
+
+    return 0;
+}
+
+static uint32_t
+control(uint32_t request, void *data, ...)
+{
+    switch(request)
+    {
+	case VOCTRL_QUERY_FORMAT:
+	    return query_format(*((uint32_t *) data));
+	default:
+	    return VO_NOTIMPL;
+    }
+}
+
+


More information about the MPlayer-dev-eng mailing list