changeset 10305:3e40b8f879c8

HM12 & NV12 "decoder" (specially interleaved YUV formats, used by Hauppauge's PVR cards) based on patch by Alex <d18c7db@hotmail.com>
author arpi
date Wed, 18 Jun 2003 00:34:35 +0000
parents 0adc47bc98aa
children b7464c68bef3
files etc/codecs.conf libmpcodecs/Makefile libmpcodecs/img_format.c libmpcodecs/img_format.h libmpcodecs/vd.c libmpcodecs/vd_hmblck.c libmpdemux/demux_rawvideo.c
diffstat 7 files changed, 165 insertions(+), 1 deletions(-) [+]
line wrap: on
line diff
--- a/etc/codecs.conf	Tue Jun 17 23:43:52 2003 +0000
+++ b/etc/codecs.conf	Wed Jun 18 00:34:35 2003 +0000
@@ -1455,6 +1455,24 @@
   fourcc yv12,YV12
   out YV12
 
+videocodec rawnv12
+  info "RAW NV12"
+  status working
+  driver hmblck
+  format 0x0 0x3231564E
+  format 0x20776172
+  fourcc nv12,NV12
+  out YV12
+
+videocodec rawhm12
+  info "RAW HM12"
+  status working
+  driver hmblck
+  format 0x0 0x32314D48
+  format 0x20776172
+  fourcc hm12,HM12
+  out YV12
+
 videocodec rawi420
   info "RAW I420"
   status working
--- a/libmpcodecs/Makefile	Tue Jun 17 23:43:52 2003 +0000
+++ b/libmpcodecs/Makefile	Wed Jun 18 00:34:35 2003 +0000
@@ -10,7 +10,7 @@
 AUDIO_SRCS=dec_audio.c ad.c $(AUDIO_SRCS_LIB) $(AUDIO_SRCS_NAT) $(AUDIO_SRCS_OPT)
 
 VIDEO_SRCS_LIB=vd_libmpeg2.c vd_nuv.c vd_lzo.c
-VIDEO_SRCS_NAT=vd_null.c vd_cinepak.c vd_qtrpza.c vd_raw.c vd_msvidc.c vd_fli.c vd_qtrle.c vd_qtsmc.c vd_roqvideo.c vd_cyuv.c vd_msrle.c vd_huffyuv.c vd_mpegpes.c vd_svq1.c vd_lcl.c vd_mtga.c vd_sgi.c
+VIDEO_SRCS_NAT=vd_null.c vd_cinepak.c vd_qtrpza.c vd_raw.c vd_hmblck.c vd_msvidc.c vd_fli.c vd_qtrle.c vd_qtsmc.c vd_roqvideo.c vd_cyuv.c vd_msrle.c vd_huffyuv.c vd_mpegpes.c vd_svq1.c vd_lcl.c vd_mtga.c vd_sgi.c
 VIDEO_SRCS_OPT=vd_realvid.c vd_ffmpeg.c vd_dshow.c vd_dmo.c vd_vfw.c vd_vfwex.c vd_odivx.c vd_divx4.c vd_xanim.c vd_xvid.c vd_libdv.c vd_qtvideo.c vd_theora.c
 VIDEO_SRCS=dec_video.c vd.c $(VIDEO_SRCS_NAT) $(VIDEO_SRCS_LIB) $(VIDEO_SRCS_OPT)
 
--- a/libmpcodecs/img_format.c	Tue Jun 17 23:43:52 2003 +0000
+++ b/libmpcodecs/img_format.c	Wed Jun 18 00:34:35 2003 +0000
@@ -32,6 +32,7 @@
 	case IMGFMT_422P: return("Planar 422P");
 	case IMGFMT_411P: return("Planar 411P");
 	case IMGFMT_NV12: return("Planar NV12");
+        case IMGFMT_HM12: return("Planar NV12 Macroblock");
 	case IMGFMT_IUYV: return("Packed IUYV");
 	case IMGFMT_IY41: return("Packed IY41");
 	case IMGFMT_IYU1: return("Packed IYU1");
--- a/libmpcodecs/img_format.h	Tue Jun 17 23:43:52 2003 +0000
+++ b/libmpcodecs/img_format.h	Wed Jun 18 00:34:35 2003 +0000
@@ -49,6 +49,7 @@
 #define IMGFMT_444P 0x50343434
 #define IMGFMT_422P 0x50323234
 #define IMGFMT_411P 0x50313134
+#define IMGFMT_HM12 0x32314D48
 
 /* Packed YUV Formats */
 
--- a/libmpcodecs/vd.c	Tue Jun 17 23:43:52 2003 +0000
+++ b/libmpcodecs/vd.c	Wed Jun 18 00:34:35 2003 +0000
@@ -39,6 +39,7 @@
 extern vd_functions_t mpcodecs_vd_odivx;
 extern vd_functions_t mpcodecs_vd_divx4;
 extern vd_functions_t mpcodecs_vd_raw;
+extern vd_functions_t mpcodecs_vd_hmblck;
 extern vd_functions_t mpcodecs_vd_xanim;
 extern vd_functions_t mpcodecs_vd_msrle;
 extern vd_functions_t mpcodecs_vd_msvidc;
@@ -89,6 +90,7 @@
 #endif
         &mpcodecs_vd_lzo,
         &mpcodecs_vd_raw,
+        &mpcodecs_vd_hmblck,
         &mpcodecs_vd_msrle,
         &mpcodecs_vd_msvidc,
         &mpcodecs_vd_fli,
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libmpcodecs/vd_hmblck.c	Wed Jun 18 00:34:35 2003 +0000
@@ -0,0 +1,138 @@
+#include <stdio.h>
+#include <stdlib.h>
+
+#include "config.h"
+#include "mp_msg.h"
+
+#include "vd_internal.h"
+
+#define TEMP_BUF_SIZE (720*576)
+
+static vd_info_t info = {
+	"Hauppauge Macroblock/NV12 Decoder",
+	"hmblck",
+	"Alex <d18c7db@hotmail.com>, A'rpi",
+	"Alex <d18c7db@hotmail.com>",
+	"uncompressed"
+};
+
+LIBVD_EXTERN(hmblck)
+
+static void de_macro_y(unsigned char* dst,unsigned char* src,int dstride,int w,int h){
+    unsigned int y;
+    // descramble Y plane
+    for (y=0; y<h; y+=16) {
+	unsigned int x;
+        for (x=0; x<w; x+=16) {
+	    unsigned int i;
+            for (i=0; i<16; i++) {
+                memcpy(dst + x + (y+i)*dstride, src, 16);
+                src+=16;
+            }
+        }
+    }
+}
+
+static void de_macro_uv(unsigned char* dstu,unsigned char* dstv,unsigned char* src,int dstride,int w,int h){
+    unsigned int y;
+    // descramble U/V plane
+    for (y=0; y<h; y+=16) {
+	unsigned int x;
+        for (x=0; x<w; x+=8) {
+	    unsigned int i;
+            for (i=0; i<16; i++) {
+		int idx=x + (y+i)*dstride;
+		dstu[idx+0]=src[0]; dstv[idx+0]=src[1];
+		dstu[idx+1]=src[2]; dstv[idx+1]=src[3];
+		dstu[idx+2]=src[4]; dstv[idx+2]=src[5];
+		dstu[idx+3]=src[6]; dstv[idx+3]=src[7];
+		dstu[idx+4]=src[8]; dstv[idx+4]=src[9];
+		dstu[idx+5]=src[10]; dstv[idx+5]=src[11];
+		dstu[idx+6]=src[12]; dstv[idx+6]=src[13];
+		dstu[idx+7]=src[14]; dstv[idx+7]=src[15];
+                src+=16;
+            }
+        }
+    }
+}
+
+/*************************************************************************
+ * convert a nv12 buffer to yv12
+ */
+static int nv12_to_yv12(void *data, int len, mp_image_t* mpi) {
+    unsigned int Y_size  = mpi->width * mpi->height;
+    unsigned int UV_size = mpi->chroma_width * mpi->chroma_height;
+    unsigned int idx;
+    unsigned char *dst_Y = mpi->planes[0];
+    unsigned char *dst_U = mpi->planes[1];
+    unsigned char *dst_V = mpi->planes[2];
+    unsigned char *src   = data + Y_size;
+
+    // sanity check raw stream
+    if ( (len != (Y_size + (UV_size<<1))) ) {
+        mp_msg(MSGT_DECVIDEO, MSGL_ERR,
+               "hmblck: Image size inconsistent with data size.\n");
+        return 0;
+    }
+    if ( (mpi->width > 720) || (mpi->height > 576) ) {
+        mp_msg(MSGT_DECVIDEO,MSGL_ERR,
+               "hmblck: Image size is too big.\n");
+        return 0;
+    }
+    if (mpi->num_planes != 3) {
+        mp_msg(MSGT_DECVIDEO,MSGL_ERR,
+               "hmblck: Incorrect number of image planes.\n");
+        return 0;
+    }
+
+    // luma data is easy, just copy it
+    memcpy(dst_Y, data, Y_size);
+
+    // chroma data is interlaced UVUV... so deinterlace it
+    for(idx=0; idx<UV_size; idx++ ) {
+        *(dst_U + idx) = *(src + (idx<<1) + 0); 
+        *(dst_V + idx) = *(src + (idx<<1) + 1);
+    }
+    return 1;
+}
+
+/*************************************************************************
+ * set/get/query special features/parameters
+ */
+static int control(sh_video_t *sh,int cmd, void *arg,...){
+    return CONTROL_UNKNOWN;
+}
+/*************************************************************************
+ * init driver
+ */
+static int init(sh_video_t *sh){
+    return mpcodecs_config_vo(sh,sh->disp_w,sh->disp_h,sh->format);
+}
+/*************************************************************************
+ * uninit driver
+ */
+static void uninit(sh_video_t *sh){
+}
+/*************************************************************************
+ * decode a frame
+ */
+static mp_image_t* decode(sh_video_t *sh,void* data,int len,int flags){
+    mp_image_t* mpi;
+
+    if(len<=0) return NULL; // skipped frame
+
+    mpi=mpcodecs_get_image(sh, MP_IMGTYPE_TEMP, MP_IMGFLAG_ACCEPT_STRIDE,
+        sh->disp_w, sh->disp_h);
+    if(!mpi) return NULL;
+
+    if(sh->format == IMGFMT_HM12) {
+        //if(!de_macro(sh, data, len, flags, mpi)) return NULL;
+	de_macro_y(mpi->planes[0],data,mpi->stride[0],mpi->w,mpi->h);
+	de_macro_uv(mpi->planes[1],mpi->planes[2],data+mpi->w*mpi->h,mpi->stride[1],mpi->w/2,mpi->h/2);
+    } else {
+	if(!nv12_to_yv12(data, len, mpi)) return NULL;
+    }
+
+    return mpi;
+}
+/*************************************************************************/
--- a/libmpdemux/demux_rawvideo.c	Tue Jun 17 23:43:52 2003 +0000
+++ b/libmpdemux/demux_rawvideo.c	Wed Jun 18 00:34:35 2003 +0000
@@ -40,6 +40,8 @@
   { "format", &format, CONF_TYPE_INT, 0, 0 , 0, NULL },
   { "y420", &format, CONF_TYPE_FLAG, 0, 0 , IMGFMT_I420, NULL },
   { "yv12", &format, CONF_TYPE_FLAG, 0, 0 , IMGFMT_YV12, NULL },
+  { "nv12", &format, CONF_TYPE_FLAG, 0, 0 , IMGFMT_NV12, NULL },
+  { "hm12", &format, CONF_TYPE_FLAG, 0, 0 , IMGFMT_HM12, NULL },
   { "yuy2", &format, CONF_TYPE_FLAG, 0, 0 , IMGFMT_YUY2, NULL },
   { "uyvy", &format, CONF_TYPE_FLAG, 0, 0 , IMGFMT_UYVY, NULL },
   { "y8", &format, CONF_TYPE_FLAG, 0, 0 , IMGFMT_Y8, NULL },
@@ -73,6 +75,8 @@
   switch(format){
   case IMGFMT_I420:
   case IMGFMT_IYUV:
+  case IMGFMT_NV12:
+  case IMGFMT_HM12:
   case IMGFMT_YV12: imgsize=width*height+2*(width>>1)*(height>>1);break;
   case IMGFMT_YUY2: imgsize=width*height*2;break;
   case IMGFMT_UYVY: imgsize=width*height*2;break;