summaryrefslogtreecommitdiffstats
path: root/gst-libs/ext/mplex/yuv4mpeg.cc
diff options
context:
space:
mode:
authorRonald S. Bultje <rbultje@ronald.bitfreak.net>2004-01-01 22:45:57 +0000
committerRonald S. Bultje <rbultje@ronald.bitfreak.net>2004-01-01 22:45:57 +0000
commit9003ed34ba8e694b44433597c890affac369c368 (patch)
treedca1fa1b1decb37f42397813935add00b4f13182 /gst-libs/ext/mplex/yuv4mpeg.cc
parent0e3024712e2a935d1f5b216cd4d7794a0132c522 (diff)
downloadgst-plugins-bad-9003ed34ba8e694b44433597c890affac369c368.tar.gz
gst-plugins-bad-9003ed34ba8e694b44433597c890affac369c368.tar.bz2
gst-plugins-bad-9003ed34ba8e694b44433597c890affac369c368.zip
configure.ac: Fix configure check for mpeg2enc. We need 1.6.1.93 instead of 1.6.1.92, since the pkg-config file of 1....
Original commit message from CVS: 2004-01-01 Ronald Bultje <rbultje@ronald.bitfreak.net> * configure.ac: Fix configure check for mpeg2enc. We need 1.6.1.93 instead of 1.6.1.92, since the pkg-config file of 1.6.1.92 is borked and it therefore uses the wrong include paths. Too bad... Note that 1.6.1.93 is not release yet. ;). Also add a check for mplex, which is now using the lib'ified mplex from mjpegtools, too. * ext/ffmpeg/gstffmpegcodecmap.c: Add codec_tag for 3ivx/xvid. For xvid, this should fix playback issues. I don't think ffmpeg handles 3ivx correctly, so this probably won't work. But it won't hurt either. * ext/ffmpeg/gstffmpegdec.c: (gst_ffmpegdec_connect), (gst_ffmpegdec_chain): * ext/ffmpeg/gstffmpegenc.c: (gst_ffmpegenc_connect), (gst_ffmpegenc_chain_audio): Fix memleak in audio encoding. Close codec if open fails, this calls the cleanup routines so we can re-use the context. * ext/mpeg2enc/gstmpeg2enc.cc: Fix pad template names/types, fix memory issue with getcaps(). * ext/mpeg2enc/gstmpeg2encoder.cc: * ext/mpeg2enc/gstmpeg2encoder.hh: Fix compile issue with new caps system (const thingy). * ext/mpeg2enc/gstmpeg2encpicturereader.cc: * ext/mpeg2enc/gstmpeg2encpicturereader.hh: We read a first frame right on initing, so that we have a caps when we init the output. This caps is cached in padprivate and read as first frame. * ext/mplex/Makefile.am: * ext/mplex/gstmplex.cc: * ext/mplex/gstmplex.h: * ext/mplex/gstmplex.hh: * ext/mplex/gstmplexibitstream.cc: * ext/mplex/gstmplexibitstream.hh: * ext/mplex/gstmplexjob.cc: * ext/mplex/gstmplexjob.hh: * ext/mplex/gstmplexoutputstream.cc: * ext/mplex/gstmplexoutputstream.hh: We wrap mjpegtools mplex. So I rewrote the plugin. The old plugin had issues, didn't do capsnego, supported only a subset of the mplex features and required a mplex fork in our local CVS. Plus that it worked agaist a very old mplex version. Rewriting was faster than updating it. * gst-libs/ext/Makefile.am: * gst-libs/ext/mplex/INSTRUCT: * gst-libs/ext/mplex/Makefile.am: * gst-libs/ext/mplex/README: * gst-libs/ext/mplex/TODO: * gst-libs/ext/mplex/ac3strm_in.cc: * gst-libs/ext/mplex/audiostrm.hh: * gst-libs/ext/mplex/audiostrm_out.cc: * gst-libs/ext/mplex/aunit.hh: * gst-libs/ext/mplex/bits.cc: * gst-libs/ext/mplex/bits.hh: * gst-libs/ext/mplex/buffer.cc: * gst-libs/ext/mplex/buffer.hh: * gst-libs/ext/mplex/fastintfns.h: * gst-libs/ext/mplex/format_codes.h: * gst-libs/ext/mplex/inputstrm.cc: * gst-libs/ext/mplex/inputstrm.hh: * gst-libs/ext/mplex/lpcmstrm_in.cc: * gst-libs/ext/mplex/mjpeg_logging.cc: * gst-libs/ext/mplex/mjpeg_logging.h: * gst-libs/ext/mplex/mjpeg_types.h: * gst-libs/ext/mplex/mpastrm_in.cc: * gst-libs/ext/mplex/mpegconsts.cc: * gst-libs/ext/mplex/mpegconsts.h: * gst-libs/ext/mplex/mplexconsts.hh: * gst-libs/ext/mplex/multplex.cc: * gst-libs/ext/mplex/outputstream.hh: * gst-libs/ext/mplex/padstrm.cc: * gst-libs/ext/mplex/padstrm.hh: * gst-libs/ext/mplex/stillsstream.cc: * gst-libs/ext/mplex/stillsstream.hh: * gst-libs/ext/mplex/systems.cc: * gst-libs/ext/mplex/systems.hh: * gst-libs/ext/mplex/vector.cc: * gst-libs/ext/mplex/vector.hh: * gst-libs/ext/mplex/videostrm.hh: * gst-libs/ext/mplex/videostrm_in.cc: * gst-libs/ext/mplex/videostrm_out.cc: * gst-libs/ext/mplex/yuv4mpeg.cc: * gst-libs/ext/mplex/yuv4mpeg.h: * gst-libs/ext/mplex/yuv4mpeg_intern.h: * gst-libs/ext/mplex/yuv4mpeg_ratio.cc: We don't fork mjpegtools' mplex in our CVS anymore. * gst/avi/gstavidemux.c: (gst_avi_demux_src_getcaps), (gst_avi_demux_add_stream): * gst/avi/gstavidemux.h: Add getcaps() function for proper caps nego. This makes some parts of AVI playback/reading work. * sys/ximage/ximagesink.c: (gst_ximagesink_sinkconnect): Resize window on new capsnego. This is probably wrong, but I'm still committing it because with current capsnego, the first successfull capsnego is auto-fixated, therefore rounded down to the lowest values in the caps. this results in a 16x16 XWindow that is not reized when real capsnego finishes. Dave, I see more cases of this, do you know a proper solution? * tools/gst-launch-ext.in: Fix MPEG-4 AAC (Apple iPod/iTunes) file commandline.
Diffstat (limited to 'gst-libs/ext/mplex/yuv4mpeg.cc')
-rw-r--r--gst-libs/ext/mplex/yuv4mpeg.cc880
1 files changed, 0 insertions, 880 deletions
diff --git a/gst-libs/ext/mplex/yuv4mpeg.cc b/gst-libs/ext/mplex/yuv4mpeg.cc
deleted file mode 100644
index a9e1aebf..00000000
--- a/gst-libs/ext/mplex/yuv4mpeg.cc
+++ /dev/null
@@ -1,880 +0,0 @@
-/*
- * yuv4mpeg.c: Functions for reading and writing "new" YUV4MPEG streams
- *
- * Copyright (C) 2001 Matthew J. Marjanovic <maddog@mir.com>
- *
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version 2
- * of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
- *
- */
-
-#include <config.h>
-
-#include <unistd.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include "yuv4mpeg.h"
-#include "yuv4mpeg_intern.h"
-#include "mjpeg_logging.h"
-
-
-static int _y4mparam_allow_unknown_tags = 1; /* default is forgiveness */
-
-static void *(*_y4m_alloc) (size_t bytes) = malloc;
-static void (*_y4m_free) (void *ptr) = free;
-
-
-
-int
-y4m_allow_unknown_tags (int yn)
-{
- int old = _y4mparam_allow_unknown_tags;
-
- if (yn >= 0)
- _y4mparam_allow_unknown_tags = (yn) ? 1 : 0;
- return old;
-}
-
-
-
-/*************************************************************************
- *
- * Convenience functions for fd read/write
- *
- * - guaranteed to transfer entire payload (or fail)
- * - returns:
- * 0 on complete success
- * +(# of remaining bytes) on eof (for y4m_read)
- * -(# of rem. bytes) on error (and ERRNO should be set)
- *
- *************************************************************************/
-
-
-ssize_t
-y4m_read (int fd, void *buf, size_t len)
-{
- ssize_t n;
- uint8_t *ptr = (uint8_t *) buf;
-
- while (len > 0) {
- n = read (fd, ptr, len);
- if (n <= 0) {
- /* return amount left to read */
- if (n == 0)
- return len; /* n == 0 --> eof */
- else
- return -len; /* n < 0 --> error */
- }
- ptr += n;
- len -= n;
- }
- return 0;
-}
-
-
-ssize_t
-y4m_write (int fd, const void *buf, size_t len)
-{
- ssize_t n;
- const uint8_t *ptr = (const uint8_t *) buf;
-
- while (len > 0) {
- n = write (fd, ptr, len);
- if (n <= 0)
- return -len; /* return amount left to write */
- ptr += n;
- len -= n;
- }
- return 0;
-}
-
-
-
-/*************************************************************************
- *
- * "Extra tags" handling
- *
- *************************************************************************/
-
-
-static char *
-y4m_new_xtag (void)
-{
- return (char *) _y4m_alloc (Y4M_MAX_XTAG_SIZE * sizeof (char));
-}
-
-
-void
-y4m_init_xtag_list (y4m_xtag_list_t * xtags)
-{
- int i;
-
- xtags->count = 0;
- for (i = 0; i < Y4M_MAX_XTAGS; i++) {
- xtags->tags[i] = NULL;
- }
-}
-
-
-void
-y4m_fini_xtag_list (y4m_xtag_list_t * xtags)
-{
- int i;
-
- for (i = 0; i < Y4M_MAX_XTAGS; i++) {
- if (xtags->tags[i] != NULL) {
- _y4m_free (xtags->tags[i]);
- xtags->tags[i] = NULL;
- }
- }
- xtags->count = 0;
-}
-
-
-void
-y4m_copy_xtag_list (y4m_xtag_list_t * dest, const y4m_xtag_list_t * src)
-{
- int i;
-
- for (i = 0; i < src->count; i++) {
- if (dest->tags[i] == NULL)
- dest->tags[i] = y4m_new_xtag ();
- strncpy (dest->tags[i], src->tags[i], Y4M_MAX_XTAG_SIZE);
- }
- dest->count = src->count;
-}
-
-
-
-static int
-y4m_snprint_xtags (char *s, int maxn, const y4m_xtag_list_t * xtags)
-{
- int i, room;
-
- for (i = 0, room = maxn - 1; i < xtags->count; i++) {
- int n = snprintf (s, room + 1, " %s", xtags->tags[i]);
-
- if ((n < 0) || (n > room))
- return Y4M_ERR_HEADER;
- s += n;
- room -= n;
- }
- s[0] = '\n'; /* finish off header with newline */
- s[1] = '\0'; /* ...and end-of-string */
- return Y4M_OK;
-}
-
-
-int
-y4m_xtag_count (const y4m_xtag_list_t * xtags)
-{
- return xtags->count;
-}
-
-
-const char *
-y4m_xtag_get (const y4m_xtag_list_t * xtags, int n)
-{
- if (n >= xtags->count)
- return NULL;
- else
- return xtags->tags[n];
-}
-
-
-int
-y4m_xtag_add (y4m_xtag_list_t * xtags, const char *tag)
-{
- if (xtags->count >= Y4M_MAX_XTAGS)
- return Y4M_ERR_XXTAGS;
- if (xtags->tags[xtags->count] == NULL)
- xtags->tags[xtags->count] = y4m_new_xtag ();
- strncpy (xtags->tags[xtags->count], tag, Y4M_MAX_XTAG_SIZE);
- (xtags->count)++;
- return Y4M_OK;
-}
-
-
-int
-y4m_xtag_remove (y4m_xtag_list_t * xtags, int n)
-{
- int i;
- char *q;
-
- if ((n < 0) || (n >= xtags->count))
- return Y4M_ERR_RANGE;
- q = xtags->tags[n];
- for (i = n; i < (xtags->count - 1); i++)
- xtags->tags[i] = xtags->tags[i + 1];
- xtags->tags[i] = q;
- (xtags->count)--;
- return Y4M_OK;
-}
-
-
-int
-y4m_xtag_clearlist (y4m_xtag_list_t * xtags)
-{
- xtags->count = 0;
- return Y4M_OK;
-}
-
-
-int
-y4m_xtag_addlist (y4m_xtag_list_t * dest, const y4m_xtag_list_t * src)
-{
- int i, j;
-
- if ((dest->count + src->count) > Y4M_MAX_XTAGS)
- return Y4M_ERR_XXTAGS;
- for (i = dest->count, j = 0; j < src->count; i++, j++) {
- if (dest->tags[i] == NULL)
- dest->tags[i] = y4m_new_xtag ();
- strncpy (dest->tags[i], src->tags[i], Y4M_MAX_XTAG_SIZE);
- }
- dest->count += src->count;
- return Y4M_OK;
-}
-
-
-/*************************************************************************
- *
- * Creators/destructors for y4m_*_info_t structures
- *
- *************************************************************************/
-
-
-void
-y4m_init_stream_info (y4m_stream_info_t * info)
-{
- if (info == NULL)
- return;
- /* initialize info */
- info->width = Y4M_UNKNOWN;
- info->height = Y4M_UNKNOWN;
- info->interlace = Y4M_UNKNOWN;
- info->framerate = y4m_fps_UNKNOWN;
- info->sampleaspect = y4m_sar_UNKNOWN;
- y4m_init_xtag_list (&(info->x_tags));
-}
-
-
-void
-y4m_copy_stream_info (y4m_stream_info_t * dest, const y4m_stream_info_t * src)
-{
- if ((dest == NULL) || (src == NULL))
- return;
- /* copy info */
- dest->width = src->width;
- dest->height = src->height;
- dest->interlace = src->interlace;
- dest->framerate = src->framerate;
- dest->sampleaspect = src->sampleaspect;
- dest->framelength = src->framelength;
- y4m_copy_xtag_list (&(dest->x_tags), &(src->x_tags));
-}
-
-
-void
-y4m_fini_stream_info (y4m_stream_info_t * info)
-{
- if (info == NULL)
- return;
- y4m_fini_xtag_list (&(info->x_tags));
-}
-
-
-void
-y4m_si_set_width (y4m_stream_info_t * si, int width)
-{
- si->width = width;
- si->framelength = (si->height * si->width) * 3 / 2;
-}
-
-int
-y4m_si_get_width (const y4m_stream_info_t * si)
-{
- return si->width;
-}
-
-void
-y4m_si_set_height (y4m_stream_info_t * si, int height)
-{
- si->height = height;
- si->framelength = (si->height * si->width) * 3 / 2;
-}
-
-int
-y4m_si_get_height (const y4m_stream_info_t * si)
-{
- return si->height;
-}
-
-void
-y4m_si_set_interlace (y4m_stream_info_t * si, int interlace)
-{
- si->interlace = interlace;
-}
-
-int
-y4m_si_get_interlace (const y4m_stream_info_t * si)
-{
- return si->interlace;
-}
-
-void
-y4m_si_set_framerate (y4m_stream_info_t * si, y4m_ratio_t framerate)
-{
- si->framerate = framerate;
-}
-
-y4m_ratio_t
-y4m_si_get_framerate (const y4m_stream_info_t * si)
-{
- return si->framerate;
-}
-
-void
-y4m_si_set_sampleaspect (y4m_stream_info_t * si, y4m_ratio_t sar)
-{
- si->sampleaspect = sar;
-}
-
-y4m_ratio_t
-y4m_si_get_sampleaspect (const y4m_stream_info_t * si)
-{
- return si->sampleaspect;
-}
-
-int
-y4m_si_get_framelength (const y4m_stream_info_t * si)
-{
- return si->framelength;
-}
-
-y4m_xtag_list_t *
-y4m_si_xtags (y4m_stream_info_t * si)
-{
- return &(si->x_tags);
-}
-
-
-
-void
-y4m_init_frame_info (y4m_frame_info_t * info)
-{
- if (info == NULL)
- return;
- /* initialize info */
- y4m_init_xtag_list (&(info->x_tags));
-}
-
-
-void
-y4m_copy_frame_info (y4m_frame_info_t * dest, const y4m_frame_info_t * src)
-{
- if ((dest == NULL) || (src == NULL))
- return;
- /* copy info */
- y4m_copy_xtag_list (&(dest->x_tags), &(src->x_tags));
-}
-
-
-void
-y4m_fini_frame_info (y4m_frame_info_t * info)
-{
- if (info == NULL)
- return;
- y4m_fini_xtag_list (&(info->x_tags));
-}
-
-
-
-/*************************************************************************
- *
- * Tag parsing
- *
- *************************************************************************/
-
-int
-y4m_parse_stream_tags (char *s, y4m_stream_info_t * i)
-{
- char *token, *value;
- char tag;
- int err;
-
- /* parse fields */
- for (token = strtok (s, Y4M_DELIM); token != NULL; token = strtok (NULL, Y4M_DELIM)) {
- if (token[0] == '\0')
- continue; /* skip empty strings */
- tag = token[0];
- value = token + 1;
- switch (tag) {
- case 'W': /* width */
- i->width = atoi (value);
- if (i->width <= 0)
- return Y4M_ERR_RANGE;
- break;
- case 'H': /* height */
- i->height = atoi (value);
- if (i->height <= 0)
- return Y4M_ERR_RANGE;
- break;
- case 'F': /* frame rate (fps) */
- if ((err = y4m_parse_ratio (&(i->framerate), value)) != Y4M_OK)
- return err;
- if (i->framerate.n < 0)
- return Y4M_ERR_RANGE;
- break;
- case 'I': /* interlacing */
- switch (value[0]) {
- case 'p':
- i->interlace = Y4M_ILACE_NONE;
- break;
- case 't':
- i->interlace = Y4M_ILACE_TOP_FIRST;
- break;
- case 'b':
- i->interlace = Y4M_ILACE_BOTTOM_FIRST;
- break;
- case '?':
- default:
- i->interlace = Y4M_UNKNOWN;
- break;
- }
- break;
- case 'A': /* sample (pixel) aspect ratio */
- if ((err = y4m_parse_ratio (&(i->sampleaspect), value)) != Y4M_OK)
- return err;
- if (i->sampleaspect.n < 0)
- return Y4M_ERR_RANGE;
- break;
- case 'X': /* 'X' meta-tag */
- if ((err = y4m_xtag_add (&(i->x_tags), token)) != Y4M_OK)
- return err;
- break;
- default:
- /* possible error on unknown options */
- if (_y4mparam_allow_unknown_tags) {
- /* unknown tags ok: store in xtag list and warn... */
- if ((err = y4m_xtag_add (&(i->x_tags), token)) != Y4M_OK)
- return err;
- mjpeg_warn ("Unknown stream tag encountered: '%s'", token);
- } else {
- /* unknown tags are *not* ok */
- return Y4M_ERR_BADTAG;
- }
- break;
- }
- }
- /* Error checking... width and height must be known since we can't
- * parse without them
- */
- if (i->width == Y4M_UNKNOWN || i->height == Y4M_UNKNOWN)
- return Y4M_ERR_HEADER;
- /* ta da! done. */
- return Y4M_OK;
-}
-
-
-
-static int
-y4m_parse_frame_tags (char *s, y4m_frame_info_t * i)
-{
- char *token, *value;
- char tag;
- int err;
-
- /* parse fields */
- for (token = strtok (s, Y4M_DELIM); token != NULL; token = strtok (NULL, Y4M_DELIM)) {
- if (token[0] == '\0')
- continue; /* skip empty strings */
- tag = token[0];
- value = token + 1;
- switch (tag) {
- case 'X': /* 'X' meta-tag */
- if ((err = y4m_xtag_add (&(i->x_tags), token)) != Y4M_OK)
- return err;
- break;
- default:
- /* possible error on unknown options */
- if (_y4mparam_allow_unknown_tags) {
- /* unknown tags ok: store in xtag list and warn... */
- if ((err = y4m_xtag_add (&(i->x_tags), token)) != Y4M_OK)
- return err;
- mjpeg_warn ("Unknown frame tag encountered: '%s'", token);
- } else {
- /* unknown tags are *not* ok */
- return Y4M_ERR_BADTAG;
- }
- break;
- }
- }
- /* ta da! done. */
- return Y4M_OK;
-}
-
-
-
-
-
-/*************************************************************************
- *
- * Read/Write stream header
- *
- *************************************************************************/
-
-
-int
-y4m_read_stream_header (int fd, y4m_stream_info_t * i)
-{
- char line[Y4M_LINE_MAX];
- char *p;
- int n;
- int err;
-
- /* read the header line */
- for (n = 0, p = line; n < Y4M_LINE_MAX; n++, p++) {
- if (read (fd, p, 1) < 1)
- return Y4M_ERR_SYSTEM;
- if (*p == '\n') {
- *p = '\0'; /* Replace linefeed by end of string */
- break;
- }
- }
- if (n >= Y4M_LINE_MAX)
- return Y4M_ERR_HEADER;
- /* look for keyword in header */
- if (strncmp (line, Y4M_MAGIC, strlen (Y4M_MAGIC)))
- return Y4M_ERR_MAGIC;
- if ((err = y4m_parse_stream_tags (line + strlen (Y4M_MAGIC), i)) != Y4M_OK)
- return err;
-
- i->framelength = (i->height * i->width) * 3 / 2;
- return Y4M_OK;
-}
-
-
-
-int
-y4m_write_stream_header (int fd, const y4m_stream_info_t * i)
-{
- char s[Y4M_LINE_MAX + 1];
- int n;
- int err;
- y4m_ratio_t rate = i->framerate;
- y4m_ratio_t aspect = i->sampleaspect;
-
- y4m_ratio_reduce (&rate);
- y4m_ratio_reduce (&aspect);
- n = snprintf (s, sizeof (s), "%s W%d H%d F%d:%d I%s A%d:%d",
- Y4M_MAGIC,
- i->width,
- i->height,
- rate.n, rate.d,
- (i->interlace == Y4M_ILACE_NONE) ? "p" :
- (i->interlace == Y4M_ILACE_TOP_FIRST) ? "t" :
- (i->interlace == Y4M_ILACE_BOTTOM_FIRST) ? "b" : "?", aspect.n, aspect.d);
- if ((n < 0) || (n > Y4M_LINE_MAX))
- return Y4M_ERR_HEADER;
- if ((err = y4m_snprint_xtags (s + n, sizeof (s) - n - 1, &(i->x_tags)))
- != Y4M_OK)
- return err;
- /* non-zero on error */
- return (y4m_write (fd, s, strlen (s)) ? Y4M_ERR_SYSTEM : Y4M_OK);
-}
-
-
-
-
-
-/*************************************************************************
- *
- * Read/Write frame header
- *
- *************************************************************************/
-
-int
-y4m_read_frame_header (int fd, y4m_frame_info_t * i)
-{
- char line[Y4M_LINE_MAX];
- char *p;
- int n;
- ssize_t remain;
-
- /* This is more clever than read_stream_header...
- Try to read "FRAME\n" all at once, and don't try to parse
- if nothing else is there...
- */
- remain = y4m_read (fd, line, sizeof (Y4M_FRAME_MAGIC) - 1 + 1); /* -'\0', +'\n' */
- if (remain < 0)
- return Y4M_ERR_SYSTEM;
- if (remain > 0) {
- /* A clean EOF should end exactly at a frame-boundary */
- if (remain == sizeof (Y4M_FRAME_MAGIC))
- return Y4M_ERR_EOF;
- else
- return Y4M_ERR_BADEOF;
- }
- if (strncmp (line, Y4M_FRAME_MAGIC, sizeof (Y4M_FRAME_MAGIC) - 1))
- return Y4M_ERR_MAGIC;
- if (line[sizeof (Y4M_FRAME_MAGIC) - 1] == '\n')
- return Y4M_OK; /* done -- no tags: that was the end-of-line. */
-
- if (line[sizeof (Y4M_FRAME_MAGIC) - 1] != Y4M_DELIM[0]) {
- return Y4M_ERR_MAGIC; /* wasn't a space -- what was it? */
- }
-
- /* proceed to get the tags... (overwrite the magic) */
- for (n = 0, p = line; n < Y4M_LINE_MAX; n++, p++) {
- if (y4m_read (fd, p, 1))
- return Y4M_ERR_SYSTEM;
- if (*p == '\n') {
- *p = '\0'; /* Replace linefeed by end of string */
- break;
- }
- }
- if (n >= Y4M_LINE_MAX)
- return Y4M_ERR_HEADER;
- /* non-zero on error */
- return y4m_parse_frame_tags (line, i);
-}
-
-
-int
-y4m_write_frame_header (int fd, const y4m_frame_info_t * i)
-{
- char s[Y4M_LINE_MAX + 1];
- int n;
- int err;
-
- n = snprintf (s, sizeof (s), "%s", Y4M_FRAME_MAGIC);
- if ((n < 0) || (n > Y4M_LINE_MAX))
- return Y4M_ERR_HEADER;
- if ((err = y4m_snprint_xtags (s + n, sizeof (s) - n - 1, &(i->x_tags)))
- != Y4M_OK)
- return err;
- /* non-zero on error */
- return (y4m_write (fd, s, strlen (s)) ? Y4M_ERR_SYSTEM : Y4M_OK);
-}
-
-
-
-/*************************************************************************
- *
- * Read/Write entire frame
- *
- *************************************************************************/
-
-int
-y4m_read_frame (int fd, const y4m_stream_info_t * si, y4m_frame_info_t * fi, uint8_t * const yuv[3])
-{
- int err;
- int w = si->width;
- int h = si->height;
-
- /* Read frame header */
- if ((err = y4m_read_frame_header (fd, fi)) != Y4M_OK)
- return err;
- /* Read luminance scanlines */
- if (y4m_read (fd, yuv[0], w * h))
- return Y4M_ERR_SYSTEM;
- /* Read chrominance scanlines */
- if (y4m_read (fd, yuv[1], w * h / 4))
- return Y4M_ERR_SYSTEM;
- if (y4m_read (fd, yuv[2], w * h / 4))
- return Y4M_ERR_SYSTEM;
-
- return Y4M_OK;
-}
-
-
-
-
-int
-y4m_write_frame (int fd, const y4m_stream_info_t * si,
- const y4m_frame_info_t * fi, uint8_t * const yuv[3])
-{
- int err;
- int w = si->width;
- int h = si->height;
-
- /* Write frame header */
- if ((err = y4m_write_frame_header (fd, fi)) != Y4M_OK)
- return err;
- /* Write luminance,chrominance scanlines */
- if (y4m_write (fd, yuv[0], w * h) ||
- y4m_write (fd, yuv[1], w * h / 4) || y4m_write (fd, yuv[2], w * h / 4))
- return Y4M_ERR_SYSTEM;
- return Y4M_OK;
-}
-
-
-
-/*************************************************************************
- *
- * Read/Write entire frame, (de)interleaved (to)from two separate fields
- *
- *************************************************************************/
-
-
-int
-y4m_read_fields (int fd, const y4m_stream_info_t * si, y4m_frame_info_t * fi,
- uint8_t * const upper_field[3], uint8_t * const lower_field[3])
-{
- int i, y, err;
- int width = si->width;
- int height = si->height;
-
- /* Read frame header */
- if ((err = y4m_read_frame_header (fd, fi)) != Y4M_OK)
- return err;
- /* Read Y', Cb, and Cr planes */
- for (i = 0; i < 3; i++) {
- uint8_t *srctop = upper_field[i];
- uint8_t *srcbot = lower_field[i];
-
- /* alternately write one line from each */
- for (y = 0; y < height; y += 2) {
- if (y4m_read (fd, srctop, width))
- return Y4M_ERR_SYSTEM;
- srctop += width;
- if (y4m_read (fd, srcbot, width))
- return Y4M_ERR_SYSTEM;
- srcbot += width;
- }
- /* for chroma, width/height are half as big */
- if (i == 0) {
- width /= 2;
- height /= 2;
- }
- }
- return Y4M_OK;
-}
-
-
-
-int
-y4m_write_fields (int fd, const y4m_stream_info_t * si,
- const y4m_frame_info_t * fi,
- uint8_t * const upper_field[3], uint8_t * const lower_field[3])
-{
- int i, y, err;
- int width = si->width;
- int height = si->height;
-
- /* Write frame header */
- if ((err = y4m_write_frame_header (fd, fi)) != Y4M_OK)
- return err;
- /* Write Y', Cb, and Cr planes */
- for (i = 0; i < 3; i++) {
- uint8_t *srctop = upper_field[i];
- uint8_t *srcbot = lower_field[i];
-
- /* alternately write one line from each */
- for (y = 0; y < height; y += 2) {
- if (y4m_write (fd, srctop, width))
- return Y4M_ERR_SYSTEM;
- srctop += width;
- if (y4m_write (fd, srcbot, width))
- return Y4M_ERR_SYSTEM;
- srcbot += width;
- }
- /* for chroma, width/height are half as big */
- if (i == 0) {
- width /= 2;
- height /= 2;
- }
- }
- return Y4M_OK;
-}
-
-
-
-/*************************************************************************
- *
- * Handy logging of stream info
- *
- *************************************************************************/
-
-void
-y4m_log_stream_info (log_level_t level, const char *prefix, const y4m_stream_info_t * i)
-{
- char s[256];
-
- snprintf (s, sizeof (s), " frame size: ");
- if (i->width == Y4M_UNKNOWN)
- snprintf (s + strlen (s), sizeof (s) - strlen (s), "(?)x");
- else
- snprintf (s + strlen (s), sizeof (s) - strlen (s), "%dx", i->width);
- if (i->height == Y4M_UNKNOWN)
- snprintf (s + strlen (s), sizeof (s) - strlen (s), "(?) pixels ");
- else
- snprintf (s + strlen (s), sizeof (s) - strlen (s), "%d pixels ", i->height);
- if (i->framelength == Y4M_UNKNOWN)
- snprintf (s + strlen (s), sizeof (s) - strlen (s), "(? bytes)");
- else
- snprintf (s + strlen (s), sizeof (s) - strlen (s), "(%d bytes)", i->framelength);
- mjpeg_log (level, "%s%s", prefix, s);
- if ((i->framerate.n == 0) && (i->framerate.d == 0))
- mjpeg_log (level, "%s frame rate: ??? fps", prefix);
- else
- mjpeg_log (level, "%s frame rate: %d/%d fps (~%f)", prefix,
- i->framerate.n, i->framerate.d, (double) i->framerate.n / (double) i->framerate.d);
- mjpeg_log (level, "%s interlace: %s", prefix,
- (i->interlace == Y4M_ILACE_NONE) ? "none/progressive" :
- (i->interlace == Y4M_ILACE_TOP_FIRST) ? "top-field-first" :
- (i->interlace == Y4M_ILACE_BOTTOM_FIRST) ? "bottom-field-first" : "anyone's guess");
- if ((i->sampleaspect.n == 0) && (i->sampleaspect.d == 0))
- mjpeg_log (level, "%ssample aspect ratio: ?:?", prefix);
- else
- mjpeg_log (level, "%ssample aspect ratio: %d:%d", prefix,
- i->sampleaspect.n, i->sampleaspect.d);
-}
-
-
-/*************************************************************************
- *
- * Convert error code to string
- *
- *************************************************************************/
-
-const char *
-y4m_strerr (int err)
-{
- switch (err) {
- case Y4M_OK:
- return "no error";
- case Y4M_ERR_RANGE:
- return "parameter out of range";
- case Y4M_ERR_SYSTEM:
- return "system error (failed read/write)";
- case Y4M_ERR_HEADER:
- return "bad stream or frame header";
- case Y4M_ERR_BADTAG:
- return "unknown header tag";
- case Y4M_ERR_MAGIC:
- return "bad header magic";
- case Y4M_ERR_XXTAGS:
- return "too many xtags";
- case Y4M_ERR_EOF:
- return "end-of-file";
- case Y4M_ERR_BADEOF:
- return "stream ended unexpectedly (EOF)";
- default:
- return "unknown error code";
- }
-}