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.
216 lines
5.0 KiB
216 lines
5.0 KiB
/*
|
|
* import_ac3.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 "import_ac3.so"
|
|
#define MOD_VERSION "v0.3.2 (2002-02-15)"
|
|
#define MOD_CODEC "(audio) AC3"
|
|
|
|
#include "transcode.h"
|
|
|
|
static int verbose_flag = TC_QUIET;
|
|
static int capability_flag = TC_CAP_PCM | TC_CAP_AC3;
|
|
|
|
#define MOD_PRE ac3
|
|
#include "import_def.h"
|
|
|
|
#include "ac3scan.h"
|
|
|
|
|
|
char import_cmd_buf[TC_BUF_MAX];
|
|
|
|
static FILE *fd;
|
|
|
|
static int codec, syncf=0;
|
|
static int pseudo_frame_size=0, real_frame_size=0, effective_frame_size=0;
|
|
static int ac3_bytes_to_go=0;
|
|
|
|
|
|
/* ------------------------------------------------------------
|
|
*
|
|
* open stream
|
|
*
|
|
* ------------------------------------------------------------*/
|
|
|
|
MOD_open
|
|
{
|
|
long sret;
|
|
|
|
// audio only
|
|
if(param->flag != TC_AUDIO) return(TC_IMPORT_ERROR);
|
|
|
|
codec = vob->im_a_codec;
|
|
syncf = vob->sync;
|
|
|
|
switch(codec) {
|
|
|
|
case CODEC_AC3:
|
|
|
|
// produce a clean sequence of AC3 frames
|
|
sret = tc_snprintf(import_cmd_buf, TC_BUF_MAX,
|
|
"tcextract -a %d -i \"%s\" -x ac3 -d %d |"
|
|
" tcextract -t raw -x ac3 -d %d",
|
|
vob->a_track, vob->audio_in_file, vob->verbose, vob->verbose);
|
|
if (sret < 0)
|
|
return(TC_IMPORT_ERROR);
|
|
|
|
if(verbose_flag) tc_log_info(MOD_NAME, "AC3->AC3");
|
|
|
|
break;
|
|
|
|
case CODEC_PCM:
|
|
|
|
if(vob->a_codec_flag==CODEC_AC3) {
|
|
|
|
sret = tc_snprintf(import_cmd_buf, TC_BUF_MAX,
|
|
"tcextract -a %d -i \"%s\" -x ac3 -d %d |"
|
|
" tcdecode -x ac3 -d %d -s %f,%f,%f -A %d",
|
|
vob->a_track, vob->audio_in_file, vob->verbose,
|
|
vob->verbose, vob->ac3_gain[0], vob->ac3_gain[1],
|
|
vob->ac3_gain[2], vob->a52_mode);
|
|
if (sret < 0)
|
|
return(TC_IMPORT_ERROR);
|
|
|
|
if(verbose_flag) tc_log_info(MOD_NAME, "AC3->PCM");
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
tc_log_warn(MOD_NAME, "invalid import codec request 0x%x", codec);
|
|
return(TC_IMPORT_ERROR);
|
|
|
|
}
|
|
|
|
// print out
|
|
if(verbose_flag)
|
|
tc_log_info(MOD_NAME, "%s", import_cmd_buf);
|
|
|
|
// set to NULL if we handle read
|
|
param->fd = NULL;
|
|
|
|
// popen
|
|
if((fd = popen(import_cmd_buf, "r"))== NULL) {
|
|
tc_log_perror(MOD_NAME, "popen pcm stream");
|
|
return(TC_IMPORT_ERROR);
|
|
}
|
|
|
|
return(TC_IMPORT_OK);
|
|
}
|
|
|
|
/* ------------------------------------------------------------
|
|
*
|
|
* decode stream
|
|
*
|
|
* ------------------------------------------------------------*/
|
|
|
|
MOD_decode
|
|
{
|
|
|
|
int ac_bytes=0, ac_off=0;
|
|
int num_frames;
|
|
|
|
// audio only
|
|
if(param->flag != TC_AUDIO) return(TC_IMPORT_ERROR);
|
|
|
|
switch(codec) {
|
|
|
|
case CODEC_AC3:
|
|
|
|
// determine frame size at the very beginning of the stream
|
|
|
|
if (pseudo_frame_size == 0) {
|
|
|
|
if (ac3scan(fd, param->buffer, param->size, &ac_off, &ac_bytes,
|
|
&pseudo_frame_size, &real_frame_size, verbose) != 0)
|
|
return(TC_IMPORT_ERROR);
|
|
|
|
} else {
|
|
ac_off = 0;
|
|
ac_bytes = pseudo_frame_size;
|
|
}
|
|
|
|
|
|
// switch to entire frames:
|
|
// bytes_to_go is the difference between requested bytes and
|
|
// delivered bytes
|
|
//
|
|
// pseudo_frame_size = average bytes per audio frame
|
|
// real_frame_size = real AC3 frame size in bytes
|
|
|
|
num_frames = (ac_bytes + ac3_bytes_to_go) / real_frame_size;
|
|
|
|
effective_frame_size = num_frames * real_frame_size;
|
|
ac3_bytes_to_go = ac_bytes + ac3_bytes_to_go - effective_frame_size;
|
|
|
|
// return effective_frame_size as physical size of audio data
|
|
param->size = effective_frame_size;
|
|
|
|
if (verbose_flag & TC_STATS)
|
|
tc_log_info(MOD_NAME, "pseudo=%d, real=%d, frames=%d, effective=%d",
|
|
ac_bytes, real_frame_size, num_frames,
|
|
effective_frame_size);
|
|
|
|
// adjust
|
|
ac_bytes=effective_frame_size;
|
|
|
|
if(syncf>0) {
|
|
//dump an ac3 frame, instead of a pcm frame
|
|
ac_bytes = real_frame_size-ac_off;
|
|
param->size = real_frame_size;
|
|
--syncf;
|
|
}
|
|
|
|
break;
|
|
|
|
case CODEC_PCM:
|
|
|
|
//default:
|
|
ac_off = 0;
|
|
ac_bytes = param->size;
|
|
break;
|
|
|
|
default:
|
|
tc_log_warn(MOD_NAME, "invalid import codec request 0x%x", codec);
|
|
return(TC_IMPORT_ERROR);
|
|
|
|
}
|
|
|
|
if (fread(param->buffer+ac_off, ac_bytes-ac_off, 1, fd) !=1)
|
|
return(TC_IMPORT_ERROR);
|
|
|
|
return(TC_IMPORT_OK);
|
|
}
|
|
|
|
/* ------------------------------------------------------------
|
|
*
|
|
* close stream
|
|
*
|
|
* ------------------------------------------------------------*/
|
|
|
|
MOD_close
|
|
{
|
|
if(param->fd != NULL) pclose(param->fd);
|
|
|
|
return(TC_IMPORT_OK);
|
|
}
|