/* * multiplex_raw.c -- write a separate plain file for each stream. * (C) 2005-2010 Francesco Romani * * 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 of the License, 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 this program. If not, see . */ #include "transcode.h" #include "libtc/optstr.h" #include "libtc/tcmodule-plugin.h" #include #include #include #include #include #include #define MOD_NAME "multiplex_raw.so" #define MOD_VERSION "v0.0.3 (2006-03-06)" #define MOD_CAP "write each stream in a separate file" #define MOD_FEATURES \ TC_MODULE_FEATURE_MULTIPLEX|TC_MODULE_FEATURE_VIDEO|TC_MODULE_FEATURE_AUDIO #define MOD_FLAGS \ TC_MODULE_FLAG_RECONFIGURABLE #define RAW_VID_EXT "vid" #define RAW_AUD_EXT "aud" static const char raw_help[] = "" "Overview:\n" " this module simply write audio and video streams in\n" " a separate plain file for each stream.\n" "Options:\n" " help produce module overview and options explanations\n"; typedef struct { int fd_aud; int fd_vid; } RawPrivateData; static int raw_inspect(TCModuleInstance *self, const char *options, const char **value) { TC_MODULE_SELF_CHECK(self, "inspect"); if (optstr_lookup(options, "help")) { *value = raw_help; } return TC_OK; } static int raw_configure(TCModuleInstance *self, const char *options, vob_t *vob) { char vid_name[PATH_MAX]; char aud_name[PATH_MAX]; RawPrivateData *pd = NULL; TC_MODULE_SELF_CHECK(self, "configure"); pd = self->userdata; // XXX if (vob->audio_out_file == NULL || !strcmp(vob->audio_out_file, "/dev/null")) { /* use affine names */ tc_snprintf(vid_name, PATH_MAX, "%s.%s", vob->video_out_file, RAW_VID_EXT); tc_snprintf(aud_name, PATH_MAX, "%s.%s", vob->video_out_file, RAW_AUD_EXT); } else { /* copy names verbatim */ strlcpy(vid_name, vob->video_out_file, PATH_MAX); strlcpy(aud_name, vob->audio_out_file, PATH_MAX); } /* avoid fd loss in case of failed configuration */ if (pd->fd_vid == -1) { pd->fd_vid = open(vid_name, O_RDWR|O_CREAT|O_TRUNC, S_IRUSR|S_IWUSR|S_IRGRP|S_IROTH); if (pd->fd_vid == -1) { tc_log_error(MOD_NAME, "failed to open video stream file"); return TC_ERROR; } } /* avoid fd loss in case of failed configuration */ if (pd->fd_aud == -1) { pd->fd_aud = open(aud_name, O_RDWR|O_CREAT|O_TRUNC, S_IRUSR|S_IWUSR|S_IRGRP|S_IROTH); if (pd->fd_aud == -1) { tc_log_error(MOD_NAME, "failed to open audio stream file"); return TC_ERROR; } } if (vob->verbose >= TC_DEBUG) { tc_log_info(MOD_NAME, "video output: %s (%s)", vid_name, (pd->fd_vid == -1) ?"FAILED" :"OK"); tc_log_info(MOD_NAME, "audio output: %s (%s)", aud_name, (pd->fd_aud == -1) ?"FAILED" :"OK"); } return TC_OK; } static int raw_stop(TCModuleInstance *self) { RawPrivateData *pd = NULL; int verr, aerr; TC_MODULE_SELF_CHECK(self, "stop"); pd = self->userdata; if (pd->fd_vid != -1) { verr = close(pd->fd_vid); if (verr) { tc_log_error(MOD_NAME, "closing video file: %s", strerror(errno)); return TC_ERROR; } pd->fd_vid = -1; } if (pd->fd_aud != -1) { aerr = close(pd->fd_aud); if (aerr) { tc_log_error(MOD_NAME, "closing audio file: %s", strerror(errno)); return TC_ERROR; } pd->fd_aud = -1; } return TC_OK; } static int raw_multiplex(TCModuleInstance *self, vframe_list_t *vframe, aframe_list_t *aframe) { ssize_t w_aud = 0, w_vid = 0; RawPrivateData *pd = NULL; TC_MODULE_SELF_CHECK(self, "multiplex"); pd = self->userdata; if (vframe != NULL && vframe->video_len > 0) { w_vid = tc_pwrite(pd->fd_vid, vframe->video_buf, vframe->video_len); if(w_vid < 0) { return TC_ERROR; } } if (aframe != NULL && aframe->audio_len > 0) { w_aud = tc_pwrite(pd->fd_aud, aframe->audio_buf, aframe->audio_len); if (w_aud < 0) { return TC_ERROR; } } return (int)(w_vid + w_aud); } static int raw_init(TCModuleInstance *self, uint32_t features) { RawPrivateData *pd = NULL; TC_MODULE_SELF_CHECK(self, "init"); TC_MODULE_INIT_CHECK(self, MOD_FEATURES, features); pd = tc_malloc(sizeof(RawPrivateData)); if (pd == NULL) { return TC_ERROR; } pd->fd_aud = -1; pd->fd_vid = -1; if (verbose) { tc_log_info(MOD_NAME, "%s %s", MOD_VERSION, MOD_CAP); } self->userdata = pd; return TC_OK; } static int raw_fini(TCModuleInstance *self) { TC_MODULE_SELF_CHECK(self, "fini"); raw_stop(self); tc_free(self->userdata); self->userdata = NULL; return TC_OK; } /*************************************************************************/ static const TCCodecID raw_codecs_in[] = { TC_CODEC_ANY, TC_CODEC_ERROR }; static const TCFormatID raw_formats_out[] = { TC_FORMAT_RAW, TC_FORMAT_ERROR }; /* a multiplexor is at the end of pipeline */ TC_MODULE_MPLEX_FORMATS_CODECS(raw); TC_MODULE_INFO(raw); static const TCModuleClass raw_class = { TC_MODULE_CLASS_HEAD(raw), .init = raw_init, .fini = raw_fini, .configure = raw_configure, .stop = raw_stop, .inspect = raw_inspect, .multiplex = raw_multiplex, }; TC_MODULE_ENTRY_POINT(raw) /*************************************************************************/ /* * Local variables: * c-file-style: "stroustrup" * c-file-offsets: ((case-label . *) (statement-case-intro . *)) * indent-tabs-mode: nil * End: * * vim: expandtab shiftwidth=4: */