| /* |
| * 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"; |
| } |
| } |