aboutsummaryrefslogtreecommitdiffhomepage
path: root/libmpdemux/demux_y4m.c
diff options
context:
space:
mode:
Diffstat (limited to 'libmpdemux/demux_y4m.c')
-rw-r--r--libmpdemux/demux_y4m.c141
1 files changed, 141 insertions, 0 deletions
diff --git a/libmpdemux/demux_y4m.c b/libmpdemux/demux_y4m.c
new file mode 100644
index 0000000000..4c96b950e3
--- /dev/null
+++ b/libmpdemux/demux_y4m.c
@@ -0,0 +1,141 @@
+// Y4M file parser by Rik Snel (using yuv4mpeg*.[ch] from
+// mjpeg.sourceforge.net) (derived from demux_viv.c)
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <string.h> /* strtok */
+
+#include "config.h"
+#include "mp_msg.h"
+#include "help_mp.h"
+#include "yuv4mpeg.h"
+
+//#include "stream.h"
+#include "demuxer.h"
+#include "stheader.h"
+#include "bswap.h"
+
+typedef struct {
+ int framenum;
+ y4m_stream_info_t* si;
+} y4m_priv_t;
+
+int y4m_check_file(demuxer_t* demuxer){
+ int orig_pos = stream_tell(demuxer->stream);
+ char buf[10];
+
+ mp_msg(MSGT_DEMUX, MSGL_V, "Checking for YUV4MPEG2\n");
+
+ stream_read(demuxer->stream, buf, 9);
+ buf[9] = 0;
+
+ if (strncmp("YUV4MPEG2", buf, 9)) {
+ mp_msg(MSGT_DEMUX, MSGL_DBG2, "Failed: YUV4MPEG2\n");
+ return 0;
+ }
+
+ mp_msg(MSGT_DEMUX,MSGL_DBG2,"Success: YUV4MPEG2\n");
+
+ stream_seek(demuxer->stream, orig_pos);
+
+return 1;
+}
+
+
+// return value:
+// 0 = EOF or no stream found
+// 1 = successfully read a packet
+int demux_y4m_fill_buffer(demuxer_t *demux) {
+ demux_stream_t *ds=demux->video;
+ demux_packet_t *dp;
+ y4m_priv_t *priv=demux->priv;
+ y4m_frame_info_t fi;
+ unsigned char *buf[3];
+ int err, size;
+
+ demux->filepos=stream_tell(demux->stream);
+
+ size = ((sh_video_t*)ds->sh)->disp_w*((sh_video_t*)ds->sh)->disp_h;
+
+ dp = new_demux_packet(3*size/2);
+
+ /* swap U and V components */
+ buf[0] = dp->buffer;
+ buf[1] = dp->buffer + 5*size/4;
+ buf[2] = dp->buffer + size;
+
+ if ((err=y4m_read_frame(demux->stream, priv->si, &fi, buf)) != Y4M_OK) {
+ mp_msg(MSGT_DEMUX, MSGL_V, "error reading frame %s\n", y4m_strerr(err));
+ return 0;
+ }
+
+ /* This seems to be the right way to calculate the presentation time stamp */
+ dp->pts=(float)priv->framenum/((sh_video_t*)ds->sh)->fps;
+ priv->framenum++;
+ dp->pos=demux->filepos;
+ dp->flags=0;
+ ds_add_packet(ds, dp);
+
+ return 1;
+}
+
+void demux_open_y4m(demuxer_t* demuxer){
+ y4m_priv_t* priv;
+ y4m_ratio_t framerate;
+ sh_video_t* sh=new_sh_video(demuxer,0);
+ int err;
+
+ demuxer->priv = malloc(sizeof(y4m_priv_t));
+ priv = demuxer->priv;
+
+ priv->framenum = 0;
+ priv->si = malloc(sizeof(y4m_stream_info_t));
+
+ y4m_init_stream_info(priv->si);
+ if ((err=y4m_read_stream_header(demuxer->stream, priv->si)) != Y4M_OK)
+ mp_msg(MSGT_DEMUXER, MSGL_FATAL, "error parsing YUV4MPEG header: %s\n", y4m_strerr(err));
+
+ sh->format = mmioFOURCC('Y', 'V', '1', '2');
+
+ if(!sh->fps) {
+ framerate = y4m_si_get_framerate(priv->si);
+ if (framerate.d != 0)
+ sh->fps=(float)framerate.n/(float)framerate.d;
+ else
+ sh->fps=15.0f;
+ }
+ sh->frametime=1.0f/sh->fps;
+
+ sh->disp_w = y4m_si_get_width(priv->si);
+ sh->disp_h = y4m_si_get_height(priv->si);
+
+ sh->bih=malloc(sizeof(BITMAPINFOHEADER));
+ memset(sh->bih,0,sizeof(BITMAPINFOHEADER));
+ sh->bih->biSize=40;
+ sh->bih->biWidth = priv->si->width;
+ sh->bih->biHeight = priv->si->height;
+ sh->bih->biPlanes=3;
+ sh->bih->biBitCount=12;
+ sh->bih->biCompression=sh->format;
+ sh->bih->biSizeImage=sh->bih->biWidth*sh->bih->biHeight*3/2; /* YV12 */
+
+ demuxer->video->sh=sh;
+ sh->ds=demuxer->video;
+ demuxer->video->id=0;
+
+ /* disable seeking, lazy */
+ demuxer->seekable = 0;
+
+ printf("YUV4MPEG2 Video stream %d size: display: %dx%d, codec: %ux%u\n",
+ demuxer->video->id, sh->disp_w, sh->disp_h, sh->bih->biWidth,
+ sh->bih->biHeight);
+}
+
+void demux_close_y4m(demuxer_t *demuxer)
+{
+ y4m_fini_stream_info(((y4m_priv_t*)demuxer->priv)->si);
+ free(((y4m_priv_t*)demuxer->priv)->si);
+ free(demuxer->priv);
+ return;
+}