# HG changeset patch # User rik # Date 1031524913 0 # Node ID ec3e58120e2aef0f69de9e35222431fe72a786a8 # Parent 45228f938e9034f66cc6764386ae036508fa2311 extensible blinkenlights driver, can currently be used for the Arcade http://www.blinkenlights.de/arcade diff -r 45228f938e90 -r ec3e58120e2a configure --- a/configure Sun Sep 08 22:36:38 2002 +0000 +++ b/configure Sun Sep 08 22:41:53 2002 +0000 @@ -200,6 +200,7 @@ --enable-tdfxfb build with tdfxfb support [disable] --enable-directfb build with DirectFB support [autodetect] --enable-zr build with ZR360[56]7/ZR36060 support [autodetect] + --enable-bl build with Blinkenlights support [disable] Audio: --disable-ossaudio disable OSS sound support [autodetect] @@ -975,6 +976,7 @@ _tdfxfb=no _directfb=auto _zr=auto +_bl=no _largefiles=no _language=en _shm=auto @@ -1130,6 +1132,8 @@ --disable-directfb) _directfb=no ;; --enable-zr) _zr=yes ;; --disable-zr) _zr=no ;; + --enable-bl) _bl=yes ;; + --disable-bl) _bl=no ;; --enable-mtrr) _mtrr=yes ;; --disable-mtrr) _mtrr=no ;; --enable-largefiles) _largefiles=yes ;; @@ -3743,6 +3747,17 @@ echores "$_zr" fi +echocheck "bl" +if test "$_bl" = yes ; then + _def_bl='#define HAVE_BL 1' + _vosrc="$_vosrc vo_bl.c" + _vomodules="bl $_vomodules" +else + _def_bl='#undef HAVE_BL' + _novomodules="bl $_novomodules" +fi +echores "$_bl" + echocheck "XviD" cat > $TMPC << EOF #include @@ -4765,6 +4780,7 @@ $_def_directfb $_def_directfb_version $_def_zr +$_def_bl $_def_mga $_def_xmga $_def_syncfb diff -r 45228f938e90 -r ec3e58120e2a libvo/video_out.c --- a/libvo/video_out.c Sun Sep 08 22:36:38 2002 +0000 +++ b/libvo/video_out.c Sun Sep 08 22:41:53 2002 +0000 @@ -81,6 +81,7 @@ extern vo_functions_t video_out_null; //extern vo_functions_t video_out_odivx; extern vo_functions_t video_out_zr; +extern vo_functions_t video_out_bl; extern vo_functions_t video_out_pgm; extern vo_functions_t video_out_md5; extern vo_functions_t video_out_syncfb; @@ -164,6 +165,9 @@ #ifdef HAVE_ZR &video_out_zr, #endif +#ifdef HAVE_BL + &video_out_bl, +#endif #ifdef HAVE_PNG &video_out_png, diff -r 45228f938e90 -r ec3e58120e2a libvo/vo_bl.c --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/libvo/vo_bl.c Sun Sep 08 22:41:53 2002 +0000 @@ -0,0 +1,420 @@ +/* + * vo_bl.c - playback using the Blinkenlights UPD protocol (and to files) + * + * UDP socket handling copied from bsender.c part of blib-0.6: + * http://sven.gimp.org/blinkenlights/ + * Copyright (c) 2001-2001 The Blinkenlights Crew: + * Sven Neumann + * Michael Natterer + * Daniel Mack + * (these portions are licensed under GNU GPL v2 or "(at your option) + * any later version") + * + * Other stuff: Copyright (C) Rik Snel 2002, License GNU GPL v2 + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "config.h" + +#include "video_out.h" +#include "video_out_internal.h" +#include "../mp_msg.h" +#include "../cfgparser.h" +#include "fastmemcpy.h" + +LIBVO_EXTERN (bl) + +static vo_info_t vo_info = +{ + "Blinkenlights driver: http://www.blinkenlights.de", + "bl", + "Rik Snel ", + "" +}; + +/* General variables */ + +static unsigned char *image = NULL; +static unsigned char *tmp = NULL; +static int framenum, yoff, stride; +static char *bl_subdevice = NULL; +static int prevpts = -1; + +typedef struct { + char *name; /* filename */ + FILE *fp; +} bl_file_t; + +typedef struct { + char *name; /* hostname */ + int port; + int fd; /* file descriptor */ +} bl_host_t; + +typedef struct { + char *name; + int img_format; + + int channels; + int width; + int height; + int bpc; /* bits per component: bpc = 3, channels = 3 => bpp = 24*/ + + /* file output functions */ + int (*init_file)(bl_file_t *file); + void (*write_frame)(bl_file_t *file, unsigned char *i, int duration); + void (*close_file)(bl_file_t *file); + + /* network output functions */ + int (*init_connection)(bl_host_t *host); + void (*send_frame)(bl_host_t *host); + void (*close_connection)(bl_host_t *host); +} bl_properties_t; + +static bl_properties_t *bl = NULL; + +/* arbitrary limit because I am too lazy to do proper memory management */ +#define BL_MAX_FILES 16 +#define BL_MAX_HOSTS 16 +static bl_file_t bl_files[BL_MAX_FILES]; +static bl_host_t bl_hosts[BL_MAX_HOSTS]; +static int no_bl_files = 0; +static int no_bl_hosts = 0; + +typedef struct { + uint32_t magic; + uint16_t height; + uint16_t width; + uint16_t channels; + uint16_t maxval; + unsigned char data[0]; +} bl_packet_t; + +static bl_packet_t *bl_packet = NULL; +static int bl_size; + +/* bml output functions */ +static int bml_init(bl_file_t *f) { + f->fp = fopen(f->name, "w"); + if (!f->fp) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: error opening %s\n", f->name); + return 1; + } + fprintf(f->fp, +"\n" +"\n" +"
\n" +" Movie autogenerated by MPlayer\n" +" http://www.mplayerhq.hu\n" +"
\n", bl->width, bl->height, bl->bpc, bl->channels); + return 0; +} + +static void bml_write_frame(bl_file_t *f, unsigned char *i, int duration) { + int j, k; + fprintf(f->fp, " \n", duration); + for (j = 0; j < bl->height; j++) { + fprintf(f->fp, " "); + for (k = 0; k < bl->width; k++) + fprintf(f->fp, "%02x", *(i + j * bl->width + k)); + fprintf(f->fp, "\n"); + } + fprintf(f->fp, " \n"); +} + +static void bml_close(bl_file_t *f) { + fprintf(f->fp, "
\n"); + fclose(f->fp); +} + +/* Blinkenlights UDP protocol */ +static int udp_init(bl_host_t *h) { + struct sockaddr_in addr; + struct hostent *dest; + + dest = gethostbyname(h->name); + if (!dest) { + mp_msg(MSGT_VO, MSGL_ERR, + "unable to resolve host %s\n", h->name); + return 1; + } + + h->fd = -1; + addr.sin_family = AF_INET; + addr.sin_port = htons(h->port); + + memcpy(&addr.sin_addr.s_addr, dest->h_addr_list[0], dest->h_length); + + h->fd = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP); + if (h->fd < 0) { + mp_msg(MSGT_VO, MSGL_ERR, + "couldn't create socket for %s\n", h->name); + return 1; + } + if (connect(h->fd, (struct sockaddr *)&addr, sizeof(addr)) < 0) { + mp_msg(MSGT_VO, MSGL_ERR, "couldn't connect socket for %s\n", + h->name); + close(h->fd); + return 1; + } + return 0; +} + +static void udp_send(bl_host_t *h) { + if (write(h->fd, bl_packet, bl_size) != bl_size) + mp_msg(MSGT_VO, MSGL_ERR, "unable to send to %s\n", h->name); +} + +static void udp_close(bl_host_t *h) { + close(h->fd); +} + +#define NO_BLS 2 + +/* currently only arcade is supported, hdl can be supported + * in principle and future projects can be supported if their + * parameters become known */ +static bl_properties_t bls[NO_BLS] = { + { "hdl", IMGFMT_BGR1, 1, 18, 8, 1, + NULL, NULL, NULL, NULL, NULL, NULL }, + { "arcade", IMGFMT_YV12, 1, 26, 20, 8, + &bml_init, &bml_write_frame, &bml_close, + &udp_init, &udp_send, &udp_close } }; + +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) +{ + framenum = 0; + if (format != IMGFMT_YV12) { + mp_msg(MSGT_VO, MSGL_ERR, "vo_bl called with wrong format"); + return 1; + } + if (width > bl->width) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: width of movie too large %d > %d\n", width, bl->width); + return 1; + } + if (height > bl->height) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: height of movie too large %d > %d\n", height, bl->height); + return 1; + } + if (!image) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: image should be initialized, internal error\n"); + return 1; + } + memset(image, 0, bl->width*bl->height*3); /* blank the image */ + mp_msg(MSGT_VO, MSGL_V, "vo_config bl called\n"); + return 0; +} + +static const vo_info_t* get_info(void) { + return &vo_info; +} + +static void draw_osd(void) { +} + +static void flip_page (void) { + int i; + + if (prevpts >= 0) for (i = 0; i < no_bl_files; i++) + bl->write_frame(&bl_files[i], tmp, (vo_pts - prevpts)/90); + memcpy(tmp, image, bl->width*bl->height*bl->channels); + prevpts = vo_pts; + + for (i = 0; i < no_bl_hosts; i++) bl->send_frame(&bl_hosts[i]); + + + framenum++; + return; +} + +static uint32_t draw_frame(uint8_t * src[]) { + int i, j; + char *source, *dest; + //printf("draw frame called\n"); +#if 0 + zr_info_t *zr = &zr_info[j]; + geo_t *g = &zr->g; + source = src[0] + 2*g->yoff*zr->vdec*zr->stride + 2*g->xoff; + dest = zr->image + 2*zr->off_y; + for (i = 0; i < g->height/zr->vdec; i++) { + memcpy(dest, source, zr->image_width*2); + dest += 2*zr->image_width; + source += zr->vdec*zr->stride; + } +#endif + return 0; +} + +static uint32_t query_format(uint32_t format) { + if (format == bl->img_format) + return VFCAP_CSP_SUPPORTED|VFCAP_CSP_SUPPORTED_BY_HW; + return 0; +} + +static void uninit(void) { + int i; + mp_msg(MSGT_VO, MSGL_V, "bl: uninit called\n"); + free(bl_packet); + bl_packet = NULL; + free(bl_subdevice); + bl_subdevice = NULL; + for (i = 0; i < no_bl_files; i++) bl->close_file(&bl_files[i]); + for (i = 0; i < no_bl_hosts; i++) bl->close_connection(&bl_hosts[i]); + no_bl_files = 0; + no_bl_hosts = 0; + bl = NULL; +} + +static void check_events(void) { +} + +static uint32_t draw_slice(uint8_t *srcimg[], int stride[], + int wf, int hf, int xf, int yf) { + int i, j, w, h, x, y; + uint8_t *dst; + uint8_t *src=srcimg[0]; + uint8_t *src1=srcimg[1]; + uint8_t *src2=srcimg[2]; + w = wf; h = hf; x = xf; y = yf; + dst=image; /* + zr->off_y + zr->image_width*(y/zr->vdec)+x;*/ + // copy Y: + for (i = 0; i < h; i++) { + memcpy(dst,src,w); + dst+=bl->width; + src+=stride[0]; + + } + return 0; +} + +static uint32_t preinit(const char *arg) { + char *p, *q; + int end = 0, i; + if (!arg || strlen(arg) == 0) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: subdevice must be given, example: -vo bl:arcade:host=localhost\n"); + return 1; + } + + bl_subdevice = malloc(strlen(arg) + 1); + if (!bl_subdevice) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: out of memory error\n"); + return 1; + } + p = bl_subdevice; + strcpy(p, arg); + mp_msg(MSGT_VO, MSGL_V, "bl: preinit called with %s\n", arg); + if (strncmp(p, "arcade", 6)) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: subdevice must start with arcade, this is the only supported output format\nat the moment, i.e. -vo bl:arcade:host=localhost\n"); + return 1; + } + bl = &bls[1]; + p += 6; + if (*p == '\0') { + no_bl_hosts = 1; + bl_hosts[0].name = "localhost"; + bl_hosts[0].port = 2323; + mp_msg(MSGT_VO, MSGL_V, "bl: no hosts/files specified, using localhost:2323\n"); + end = 1; + } else if (*p != ':') { + mp_msg(MSGT_VO, MSGL_ERR, "bl: syntax error in subdevice\n"); + return 1; + } + p++; + + while (!end) { + q = p + 5; + if (!strncmp(p, "file=", 5)) { + if (no_bl_files == BL_MAX_FILES) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: maximum number of hosts reached (%d)\n", BL_MAX_FILES); + return 1; + } + p += 5; + while (*q != ',' && *q != '\0') q++; + if (*q == '\0') end = 1; + *q = '\0'; + bl_files[no_bl_files].name = p; + mp_msg(MSGT_VO, MSGL_V, "blfile[%d]: %s\n", + no_bl_files, p); + no_bl_files++; + } else if (!strncmp(p, "host=", 5)) { + if (no_bl_hosts == BL_MAX_HOSTS) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: maximum number of hosts reached (%d)\n", BL_MAX_HOSTS); + return 1; + } + p += 5; + while (*q != ',' && *q != '\0' && *q != ':') q++; + if (*q == ':') { + *q++ = '\0'; + bl_hosts[no_bl_hosts].name = p; + bl_hosts[no_bl_hosts].port = atoi(q); + while (*q != ',' && *q != '\0') q++; + if (*q == '\0') end = 1; + } else { + /* use default port */ + if (*q == '\0') end = 1; + *q = '\0'; + bl_hosts[no_bl_hosts].name = p; + bl_hosts[no_bl_hosts].port = 2323; + } + mp_msg(MSGT_VO, MSGL_V, + "blhost[%d]: %s:%d\n", + no_bl_hosts, p, + bl_hosts[no_bl_hosts].port); + no_bl_hosts++; + } else { + mp_msg(MSGT_VO, MSGL_ERR, "bl: syntax error in entry %d in subdevice %s, should be a comma seperated\nlist of host=name:port and file=foo.bml\n", no_bl_hosts, no_bl_files, arg); + return 1; + } + p = ++q; + } + + bl_size = bl->width*bl->height*bl->channels + 12; + /* enough space for RGB 24 bit + header */ + bl_packet = malloc(bl->width*bl->height*3+12); + image = ((unsigned char*)bl_packet + 12); + tmp = malloc(bl->width*bl->height*bl->channels); + + if (!bl_packet || !tmp) { + mp_msg(MSGT_VO, MSGL_ERR, "bl: out of memory error\n"); + return 1; + } + bl_packet->magic = htonl(0x23542666); + bl_packet->width = htons(bl->width); + bl_packet->height = htons(bl->height); + bl_packet->channels = htons(bl->channels); + bl_packet->maxval = htons(2<bpc - 1); + + /* open all files */ + for (i = 0; i < no_bl_files; i++) + if (bl->init_file(&bl_files[i])) return 1; + + /* open all sockets */ + for (i = 0; i < no_bl_hosts; i++) + if (bl->init_connection(&bl_hosts[i])) return 1; + + + return 0; +} + +static uint32_t control(uint32_t request, void *data, ...) { + switch (request) { + case VOCTRL_QUERY_FORMAT: + return query_format(*((uint32_t*)data)); + } + return VO_NOTIMPL; +}