mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-27 17:48:26 +00:00
881 lines
20 KiB
C++
881 lines
20 KiB
C++
|
/*
|
||
|
* 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";
|
||
|
}
|
||
|
}
|