You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
335 lines
9.8 KiB
335 lines
9.8 KiB
4 years ago
|
/*
|
||
|
* export_dvraw.c
|
||
|
*
|
||
|
* Copyright (C) Thomas Oestreich - June 2001
|
||
|
*
|
||
|
* 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.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#define MOD_NAME "export_dvraw.so"
|
||
|
#define MOD_VERSION "v0.4.1 (2007-08-17)"
|
||
|
#define MOD_CODEC "(video) Digital Video | (audio) PCM"
|
||
|
|
||
|
#include "transcode.h"
|
||
|
#include "libtc/libtc.h"
|
||
|
#include "libtc/optstr.h"
|
||
|
#include "libtcvideo/tcvideo.h"
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <libdv/dv.h>
|
||
|
|
||
|
static int verbose_flag=TC_QUIET;
|
||
|
static int capability_flag=TC_CAP_PCM|TC_CAP_RGB|TC_CAP_YUV|TC_CAP_VID|TC_CAP_YUV422;
|
||
|
|
||
|
#define MOD_PRE dvraw
|
||
|
#include "export_def.h"
|
||
|
|
||
|
static int fd = -1;
|
||
|
|
||
|
/* only 2 channesl supported */
|
||
|
#define MAX_CHANNELS 2
|
||
|
static int16_t *audio_bufs[MAX_CHANNELS] = { NULL, NULL };
|
||
|
|
||
|
static uint8_t *target = NULL, *vbuf = NULL;
|
||
|
|
||
|
static dv_encoder_t *encoder = NULL;
|
||
|
static uint8_t *pixels[3] = { NULL, NULL, NULL }, *tmp_buf = NULL;
|
||
|
static TCVHandle tcvhandle;
|
||
|
|
||
|
static int frame_size = 0, format = 0;
|
||
|
static int pass_through = 0;
|
||
|
|
||
|
static int chans = 0, rate = 0;
|
||
|
static int dv_yuy2_mode = 0;
|
||
|
static int dv_uyvy_mode = 0;
|
||
|
static int is_PAL = 0;
|
||
|
|
||
|
/* ------------------------------------------------------------
|
||
|
*
|
||
|
* init codec
|
||
|
*
|
||
|
* ------------------------------------------------------------*/
|
||
|
|
||
|
MOD_init
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
if (param->flag == TC_VIDEO) {
|
||
|
is_PAL = (vob->ex_v_height == PAL_H);
|
||
|
target = tc_bufalloc(TC_FRAME_DV_PAL);
|
||
|
vbuf = tc_bufalloc(PAL_W * PAL_H * 3);
|
||
|
|
||
|
tcvhandle = tcv_init();
|
||
|
|
||
|
if (vob->dv_yuy2_mode == 1) {
|
||
|
tmp_buf = tc_bufalloc(PAL_W * PAL_H * 2); //max frame
|
||
|
dv_yuy2_mode = 1;
|
||
|
}
|
||
|
|
||
|
if (vob->im_v_codec == CODEC_YUV422) {
|
||
|
tmp_buf = tc_bufalloc(PAL_W * PAL_H * 2); //max frame
|
||
|
dv_uyvy_mode = 1;
|
||
|
}
|
||
|
|
||
|
encoder = dv_encoder_new(FALSE, FALSE, FALSE);
|
||
|
return TC_OK;
|
||
|
}
|
||
|
|
||
|
if (param->flag == TC_AUDIO) {
|
||
|
// tmp audio buffer
|
||
|
for ( i = 0; i < MAX_CHANNELS; i++) {
|
||
|
audio_bufs[i] = tc_malloc(DV_AUDIO_MAX_SAMPLES * sizeof(int16_t));
|
||
|
if (!(audio_bufs[i])) {
|
||
|
tc_log_error(MOD_NAME, "out of memory");
|
||
|
return TC_ERROR;
|
||
|
}
|
||
|
}
|
||
|
return TC_OK;
|
||
|
}
|
||
|
|
||
|
return(TC_ERROR);
|
||
|
}
|
||
|
|
||
|
/* ------------------------------------------------------------
|
||
|
*
|
||
|
* open outputfile
|
||
|
*
|
||
|
* ------------------------------------------------------------*/
|
||
|
|
||
|
MOD_open
|
||
|
{
|
||
|
int bytealignment = 0, bytespersecond = 0, bytesperframe = 0;
|
||
|
|
||
|
if (param->flag == TC_VIDEO) {
|
||
|
fd = open(vob->video_out_file, O_RDWR|O_CREAT|O_TRUNC,
|
||
|
S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH);
|
||
|
if (fd < 0) {
|
||
|
tc_log_perror(MOD_NAME, "open file");
|
||
|
return TC_ERROR;
|
||
|
}
|
||
|
|
||
|
switch (vob->im_v_codec) {
|
||
|
case CODEC_RGB:
|
||
|
format = 0;
|
||
|
if (verbose >= TC_DEBUG)
|
||
|
tc_log_info(MOD_NAME, "raw format is RGB");
|
||
|
break;
|
||
|
case CODEC_YUV:
|
||
|
format = 1;
|
||
|
if (verbose >= TC_DEBUG)
|
||
|
tc_log_info(MOD_NAME, "raw format is YUV420P");
|
||
|
break;
|
||
|
case CODEC_YUV422:
|
||
|
format = 2;
|
||
|
if (verbose >= TC_DEBUG)
|
||
|
tc_log_info(MOD_NAME, "raw format is YUV422");
|
||
|
break;
|
||
|
case CODEC_RAW:
|
||
|
case CODEC_RAW_YUV:
|
||
|
format = 1;
|
||
|
pass_through = 1;
|
||
|
break;
|
||
|
default:
|
||
|
tc_log_warn(MOD_NAME, "codec not supported");
|
||
|
return TC_ERROR;
|
||
|
}
|
||
|
|
||
|
frame_size = (is_PAL) ?TC_FRAME_DV_PAL :TC_FRAME_DV_NTSC;
|
||
|
|
||
|
if (verbose >= TC_DEBUG)
|
||
|
tc_log_info(MOD_NAME, "encoding to %s DV",
|
||
|
(is_PAL) ?"PAL" :"NTSC");
|
||
|
|
||
|
// Store aspect ratio - ex_asr uses the value 3 for 16x9 (XXX: tricky)
|
||
|
encoder->is16x9 = (((vob->ex_asr < 0) ?vob->im_asr :vob->ex_asr) == 3);
|
||
|
encoder->isPAL = is_PAL;
|
||
|
encoder->vlc_encode_passes = 3;
|
||
|
encoder->static_qno = 0;
|
||
|
if (vob->ex_v_string != NULL)
|
||
|
if (optstr_get(vob->ex_v_string, "qno", "%d", &encoder->static_qno) == 1)
|
||
|
tc_log_info(MOD_NAME, "using quantisation: %d", encoder->static_qno);
|
||
|
encoder->force_dct = DV_DCT_AUTO;
|
||
|
|
||
|
return TC_OK;
|
||
|
}
|
||
|
|
||
|
if (param->flag == TC_AUDIO) {
|
||
|
if (!encoder) {
|
||
|
tc_log_warn(MOD_NAME, "-y XXX,dvraw is not possible without the video");
|
||
|
tc_log_warn(MOD_NAME, "export module also being dvraw");
|
||
|
return TC_ERROR;
|
||
|
}
|
||
|
chans = vob->dm_chan;
|
||
|
//re-sampling only with -J resample possible
|
||
|
rate = vob->a_rate;
|
||
|
|
||
|
bytealignment = (chans == 2) ?4 :2;
|
||
|
bytespersecond = rate * bytealignment;
|
||
|
bytesperframe = bytespersecond/(is_PAL ?25 :30);
|
||
|
|
||
|
if(verbose >= TC_DEBUG)
|
||
|
tc_log_info(MOD_NAME, "audio: CH=%d, f=%d, balign=%d, bps=%d, bpf=%d",
|
||
|
chans, rate, bytealignment, bytespersecond, bytesperframe);
|
||
|
|
||
|
return TC_OK;
|
||
|
}
|
||
|
return TC_ERROR;
|
||
|
}
|
||
|
|
||
|
/* ------------------------------------------------------------
|
||
|
*
|
||
|
* encode and export
|
||
|
*
|
||
|
* ------------------------------------------------------------*/
|
||
|
|
||
|
MOD_encode
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
if (param->flag == TC_VIDEO) {
|
||
|
if (pass_through) {
|
||
|
ac_memcpy(target, param->buffer, frame_size);
|
||
|
} else {
|
||
|
ac_memcpy(vbuf, param->buffer, param->size);
|
||
|
}
|
||
|
|
||
|
return TC_OK;
|
||
|
}
|
||
|
|
||
|
if (param->flag == TC_AUDIO) {
|
||
|
int16_t *abufs[2] = { audio_bufs[0], audio_bufs[1] }; /* working copies */
|
||
|
time_t now = time(NULL);
|
||
|
int achans = chans;
|
||
|
|
||
|
if (!pass_through) {
|
||
|
if (dv_uyvy_mode) {
|
||
|
tcv_convert(tcvhandle,
|
||
|
vbuf, tmp_buf, PAL_W, (encoder->isPAL) ? PAL_H : NTSC_H,
|
||
|
(format==2) ? IMG_YUV422P : IMG_YUV420P, IMG_UYVY);
|
||
|
pixels[0] = pixels[1] = pixels[2] = tmp_buf;
|
||
|
} else if (dv_yuy2_mode) {
|
||
|
tcv_convert(tcvhandle,
|
||
|
vbuf, tmp_buf, PAL_W, (encoder->isPAL) ? PAL_H : NTSC_H,
|
||
|
(format==2) ? IMG_YUV422P : IMG_YUV420P, IMG_YUY2);
|
||
|
pixels[0] = pixels[1] = pixels[2] = tmp_buf;
|
||
|
} else {
|
||
|
pixels[0] = vbuf;
|
||
|
if (encoder->isPAL) {
|
||
|
pixels[1] = pixels[0] + PAL_W*PAL_H;
|
||
|
pixels[2] = pixels[1] + (PAL_W/2)*(format==2?PAL_H:PAL_H/2);
|
||
|
} else {
|
||
|
pixels[1] = pixels[0] + NTSC_W*NTSC_H;
|
||
|
pixels[2] = pixels[1] + (NTSC_W/2)*(format==2?NTSC_H:NTSC_H/2);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
dv_encode_full_frame(encoder, pixels, (format)?e_dv_color_yuv:e_dv_color_rgb, target);
|
||
|
} //no pass-through
|
||
|
/*
|
||
|
* samples * channels * bits = size;
|
||
|
* so
|
||
|
* samples = size / (channels * bits)
|
||
|
*/
|
||
|
encoder->samples_this_frame = param->size / (chans * sizeof(int16_t));
|
||
|
dv_encode_metadata(target, encoder->isPAL, encoder->is16x9, &now, 0);
|
||
|
dv_encode_timecode(target, encoder->isPAL, 0);
|
||
|
|
||
|
#ifdef WORDS_BIGENDIAN
|
||
|
for (i=0; i<param->size; i+=2) {
|
||
|
uint8_t tmp = param->buffer[i];
|
||
|
param->buffer[i] = param->buffer[i+1];
|
||
|
param->buffer[i+1] = tmp;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
// Although dv_encode_full_audio supports 4 channels, the internal
|
||
|
// PCM data (param->buffer) is only carrying 2 channels so only deal
|
||
|
// with 1 or 2 channel audio.
|
||
|
// Work around apparent bug in dv_encode_full_audio when chans == 1
|
||
|
// by putting silence in 2nd channel and calling with chans = 2
|
||
|
if (chans == 1) {
|
||
|
abufs[0] = (int16_t *)param->buffer; /* avoid ac_memcpy */
|
||
|
memset(abufs[1], 0, DV_AUDIO_MAX_SAMPLES * 2);
|
||
|
achans = 2;
|
||
|
} else {
|
||
|
// assume 2 channel, demultiplex for libdv API
|
||
|
for(i = 0; i < param->size/4; i++) { // XXX magic number
|
||
|
abufs[0][i] = ((int16_t *)param->buffer)[i * 2 ];
|
||
|
abufs[1][i] = ((int16_t *)param->buffer)[i * 2 + 1];
|
||
|
}
|
||
|
}
|
||
|
dv_encode_full_audio(encoder, abufs, achans, rate, target);
|
||
|
|
||
|
if (tc_pwrite(fd, target, frame_size) != frame_size) {
|
||
|
tc_log_perror(MOD_NAME, "write frame");
|
||
|
return TC_ERROR;
|
||
|
}
|
||
|
|
||
|
return TC_OK;
|
||
|
}
|
||
|
|
||
|
return TC_ERROR;
|
||
|
}
|
||
|
|
||
|
/* ------------------------------------------------------------
|
||
|
*
|
||
|
* stop encoder
|
||
|
*
|
||
|
* ------------------------------------------------------------*/
|
||
|
|
||
|
MOD_stop
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
if (param->flag == TC_VIDEO) {
|
||
|
dv_encoder_free(encoder);
|
||
|
tcv_free(tcvhandle);
|
||
|
return TC_OK;
|
||
|
}
|
||
|
|
||
|
if (param->flag == TC_AUDIO) {
|
||
|
for(i = 0; i < MAX_CHANNELS; i++)
|
||
|
tc_free(audio_bufs[i]);
|
||
|
return TC_OK;
|
||
|
}
|
||
|
|
||
|
return TC_ERROR;
|
||
|
}
|
||
|
|
||
|
/* ------------------------------------------------------------
|
||
|
*
|
||
|
* close outputfiles
|
||
|
*
|
||
|
* ------------------------------------------------------------*/
|
||
|
|
||
|
MOD_close
|
||
|
{
|
||
|
if(param->flag == TC_VIDEO) {
|
||
|
close(fd);
|
||
|
return TC_OK;
|
||
|
}
|
||
|
|
||
|
if (param->flag == TC_AUDIO)
|
||
|
return TC_OK;
|
||
|
|
||
|
return TC_ERROR;
|
||
|
}
|
||
|
|
||
|
|