diff options
Diffstat (limited to 'debian/transcode/transcode-1.1.7/import/extract_yuv.c')
| -rw-r--r-- | debian/transcode/transcode-1.1.7/import/extract_yuv.c | 276 |
1 files changed, 276 insertions, 0 deletions
diff --git a/debian/transcode/transcode-1.1.7/import/extract_yuv.c b/debian/transcode/transcode-1.1.7/import/extract_yuv.c new file mode 100644 index 00000000..eedbcc5f --- /dev/null +++ b/debian/transcode/transcode-1.1.7/import/extract_yuv.c @@ -0,0 +1,276 @@ +/* + * extract_yuv.c + * + * Copyright (C) Thomas Oestreich - June 2001 + * Copyright (C) Francesco Romani - March 2006 + * + * This file is part of transcode, a video stream processing tool + * + * transcode 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, or (at your option) + * any later version. + * + * transcode 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 GNU Make; see the file COPYING. If not, write to + * the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. + * + */ + +#include "config.h" + +#include "transcode.h" +#include "framebuffer.h" +#include "tcinfo.h" +#include "libtc/libtc.h" +#include "libtc/ratiocodes.h" +#include "libtc/tcframes.h" + +#include "ioaux.h" +#include "avilib/avilib.h" +#include "tc.h" + +#if defined HAVE_MJPEGTOOLS +/* assert using new code (FIXME?) */ + +#if defined(HAVE_MJPEGTOOLS_INC) +#include "yuv4mpeg.h" +#include "mpegconsts.h" +#else +#include "mjpegtools/yuv4mpeg.h" +#include "mjpegtools/mpegconsts.h" +#endif + +/* ------------------------------------------------------------ + * + * yuv extract thread + * + * magic: TC_MAGIC_YUV4MPEG + * TC_MAGIC_RAW <-- default + * + * + * ------------------------------------------------------------*/ + + +static int extract_yuv_y4m(info_t *ipipe) +{ + vframe_list_t *vptr = NULL; + uint8_t *planes[3]; + int planesize[3]; + int ch_mode, w, h, ret = 0, i = 0, errnum; + + y4m_frame_info_t frameinfo; + y4m_stream_info_t streaminfo; + + /* initialize stream-information */ + y4m_accept_extensions(1); + y4m_init_stream_info(&streaminfo); + y4m_init_frame_info(&frameinfo); + + errnum = y4m_read_stream_header(ipipe->fd_in, &streaminfo); + if (errnum != Y4M_OK) { + tc_log_error(__FILE__, "Couldn't read YUV4MPEG header: %s!", + y4m_strerr (errnum)); + return 1; + } + if (y4m_si_get_plane_count(&streaminfo) != 3) { + tc_log_error(__FILE__, "Only 3-plane formats supported"); + return 1; + } + ch_mode = y4m_si_get_chroma(&streaminfo); + if (ch_mode != Y4M_CHROMA_420JPEG && ch_mode != Y4M_CHROMA_420MPEG2 + && ch_mode != Y4M_CHROMA_420PALDV) { + tc_log_error(__FILE__, "sorry, chroma mode `%s' (%i) not supported", + y4m_chroma_description(ch_mode), ch_mode); + return 1; + } + + w = y4m_si_get_width(&streaminfo); + h = y4m_si_get_height(&streaminfo); + vptr = tc_new_video_frame(w, h, TC_CODEC_YUV420P, TC_TRUE); + + if (!vptr) { + tc_log_error(__FILE__, "can't allocate buffer (%ix%i)", w, h); + return 1; + } + planes[0] = vptr->video_buf_Y[0]; + planes[1] = vptr->video_buf_U[0]; + planes[2] = vptr->video_buf_V[0]; + + planesize[0] = y4m_si_get_plane_length(&streaminfo, 0); + planesize[1] = y4m_si_get_plane_length(&streaminfo, 1); + planesize[2] = y4m_si_get_plane_length(&streaminfo, 2); + + while (1) { + errnum = y4m_read_frame(ipipe->fd_in, &streaminfo, &frameinfo, planes); + if (errnum != Y4M_OK) { + break; + } + for (i = 0; i < 3; i++) { + ret = tc_pwrite(ipipe->fd_out, planes[i], planesize[i]); + if (ret != planesize[i]) { + tc_log_perror(__FILE__, "error while writing output data"); + break; + } + } + } + + tc_del_video_frame(vptr); + y4m_fini_frame_info(&frameinfo); + y4m_fini_stream_info(&streaminfo); + + return 0; +} + +static int extract_yuv_avi(info_t *ipipe) +{ + avi_t *avifile=NULL; + char *video; + + int key; + long frames, bytes, n; + + if (ipipe->nav_seek_file) { + avifile = AVI_open_indexfd(ipipe->fd_in, 0, ipipe->nav_seek_file); + } else { + avifile = AVI_open_fd(ipipe->fd_in, 1); + } + if (NULL == avifile) { + AVI_print_error("AVI open"); + return 1; + } + + // read video info; + frames = AVI_video_frames(avifile); + if (ipipe->frame_limit[1] < frames) { + frames=ipipe->frame_limit[1]; + } + + if (ipipe->verbose & TC_STATS) { + tc_log_info(__FILE__, "%ld video frames", frames); + } + // allocate space, assume max buffer size + video = tc_bufalloc(SIZE_RGB_FRAME); + if (video == NULL) { + tc_log_error(__FILE__, "out of memory"); + return 1; + } + + AVI_set_video_position(avifile, ipipe->frame_limit[0]); + for (n = ipipe->frame_limit[0]; n <= frames; ++n) { + bytes = AVI_read_frame(avifile, video, &key); + if (bytes < 0) { + return 1; + } + if (tc_pwrite(ipipe->fd_out, video, bytes) != bytes) { + tc_log_perror(__FILE__, "error while writing output data"); + return 1; + } + } + tc_buffree(video); + + return 0; +} + +static int extract_yuv_raw(info_t *ipipe) +{ + if (ipipe->magic == TC_MAGIC_UNKNOWN) { + tc_log_warn(__FILE__, "no file type specified, assuming (%s)", + filetype(TC_MAGIC_RAW)); + } + return tc_preadwrite(ipipe->fd_in, ipipe->fd_out); +} + +void extract_yuv(info_t *ipipe) +{ + int error = 0; + + switch(ipipe->magic) { + case TC_MAGIC_YUV4MPEG: + error = extract_yuv_y4m(ipipe); + break; + case TC_MAGIC_AVI: + error = extract_yuv_avi(ipipe); + break; + case TC_MAGIC_RAW: + default: + error = extract_yuv_raw(ipipe); + break; + } + if (error) { + tc_log_error(__FILE__, "write failed"); + import_exit(error); + } +} + +void probe_yuv(info_t *ipipe) +{ + int errnum = Y4M_OK; + y4m_frame_info_t frameinfo; + y4m_stream_info_t streaminfo; + y4m_ratio_t r; + + /* initialize stream-information */ + y4m_accept_extensions(1); + y4m_init_stream_info(&streaminfo); + y4m_init_frame_info(&frameinfo); + + errnum = y4m_read_stream_header(ipipe->fd_in, &streaminfo); + if (errnum != Y4M_OK) { + tc_log_error(__FILE__, "Couldn't read YUV4MPEG header: %s!", + y4m_strerr(errnum)); + import_exit(1); + } + + ipipe->probe_info->width = y4m_si_get_width(&streaminfo); + ipipe->probe_info->height = y4m_si_get_height(&streaminfo); + + r = y4m_si_get_framerate(&streaminfo); + ipipe->probe_info->fps = (double)r.n / (double)r.d; + tc_frc_code_from_ratio(&(ipipe->probe_info->frc), r.n, r.d); + + r = y4m_si_get_sampleaspect(&streaminfo); + tc_asr_code_from_ratio(&(ipipe->probe_info->asr), r.n, r.d); + + ipipe->probe_info->codec=TC_CODEC_YUV420P; + ipipe->probe_info->magic=TC_MAGIC_YUV4MPEG; + + y4m_fini_frame_info(&frameinfo); + y4m_fini_stream_info(&streaminfo); +} + +#else /* HAVE_MJPEGTOOLS */ + +void extract_yuv(info_t *ipipe) +{ + tc_log_error(__FILE__, "No support for YUV4MPEG compiled in."); + tc_log_error(__FILE__, "Recompile with mjpegtools support enabled."); + import_exit(1); +} + +void probe_yuv(info_t * ipipe) +{ + tc_log_error(__FILE__, "No support for YUV4MPEG compiled in."); + tc_log_error(__FILE__, "Recompile with mjpegtools support enabled."); + ipipe->probe_info->codec = TC_CODEC_UNKNOWN; + ipipe->probe_info->magic = TC_MAGIC_UNKNOWN; +} + +#endif /* HAVE_MJPEGTOOLS */ + +/*************************************************************************/ + +/* + * Local variables: + * c-file-style: "stroustrup" + * c-file-offsets: ((case-label . *) (statement-case-intro . *)) + * indent-tabs-mode: nil + * End: + * + * vim: expandtab shiftwidth=4: + */ |
